mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-06-09 15:07:49 +00:00
Compare commits
160 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| fecbcab48d | |||
| 2da476b276 | |||
| 3232fc90bb | |||
| ef985b5da5 | |||
| 55666c1dcd | |||
| c107f0f683 | |||
| ba8f00a560 | |||
| 65a1282312 | |||
| 1ca632d838 | |||
| f0403e2699 | |||
| 3e34da7385 | |||
| 8750e2ffec | |||
| 3429529162 | |||
| 32b8acb9dc | |||
| 1992005b01 | |||
| 0d7a0e30cd | |||
| b12f563bc4 | |||
| d0e7b7aad1 | |||
| c676aed371 | |||
| e781115390 | |||
| 7bbb6a836c | |||
| 6bbab2f1d5 | |||
| 70546c2302 | |||
| 6c7d8ac83e | |||
| 1e48bc8762 | |||
| 77a30837e2 | |||
| a63460e853 | |||
| 1be0fa1a5f | |||
| c9e3905a65 | |||
| 495f5f5cc1 | |||
| 71e80634b1 | |||
| af2b3366ba | |||
| e015e20b5c | |||
| d92d28c892 | |||
| 60bdd4a31a | |||
| cce0d44f83 | |||
| c8623e2f7c | |||
| 6fc4ad5773 | |||
| e61625d2f7 | |||
| 907b695526 | |||
| 621d0d2bfd | |||
| 32c34be2c6 | |||
| d2aa3ebac7 | |||
| 02b86be4c5 | |||
| 99e185a16a | |||
| 995d585f54 | |||
| d05be1aef4 | |||
| 872a71efda | |||
| 6a2d6eec92 | |||
| 18f9da1eaf | |||
| 45bbe2ab1b | |||
| 18bd8ad1d9 | |||
| a40adc22d2 | |||
| 04626560eb | |||
| 945683bdf5 | |||
| 490c7fc9fd | |||
| 4f03b631ef | |||
| eac671fa80 | |||
| 1fd3f96038 | |||
| 845c14cf95 | |||
| a04ae14271 | |||
| cf0fc9e7e6 | |||
| 0caad642be | |||
| a76ab69a84 | |||
| e940c30050 | |||
| aab1c3a2d5 | |||
| f9226a0b41 | |||
| bd8b149f11 | |||
| 68a132f4bb | |||
| e576b73049 | |||
| e138547f3b | |||
| e7038a394b | |||
| e0b9192e91 | |||
| 748375b269 | |||
| 09bc31c602 | |||
| 6474dd57b3 | |||
| ececc5e5e1 | |||
| d9041a89bb | |||
| 83aa23f677 | |||
| 3d2bc11058 | |||
| c328b9a880 | |||
| e29ceffa37 | |||
| 8f7762f769 | |||
| 4c31587771 | |||
| ecac927583 | |||
| 126294d288 | |||
| aa5d4ab818 | |||
| c75281ebd9 | |||
| 06070871c7 | |||
| 5511dc78dc | |||
| 29f8b4fbdd | |||
| 4c6942c7b7 | |||
| 699e4448d1 | |||
| 6f02387d69 | |||
| fc5d5d8ad4 | |||
| 0d6ab5e718 | |||
| 547fc309f5 | |||
| 094e8f904b | |||
| 0d928d5856 | |||
| 0357531e73 | |||
| 2fa279c073 | |||
| 90ed035abd | |||
| 81b9e88bb6 | |||
| 7e01d8afb9 | |||
| d5a98b95dc | |||
| 8900937e71 | |||
| cad80e2216 | |||
| 34dad85734 | |||
| b1d2d30773 | |||
| ca3ad810c6 | |||
| f910d165e4 | |||
| 6f6985ef32 | |||
| d07f074fb1 | |||
| 561481679e | |||
| d86577f007 | |||
| 49ea3f0295 | |||
| 7bab8c66c1 | |||
| f9de4d28f9 | |||
| 45028212e1 | |||
| 0f285dc684 | |||
| d0cd82c703 | |||
| 48351661c5 | |||
| 19d7cd33ca | |||
| 03474dc1fa | |||
| 916fa4205b | |||
| b2be67e052 | |||
| 0ac40adb5a | |||
| 655b1ad5fe | |||
| cee59c2b99 | |||
| da99bf1429 | |||
| 28fd711ece | |||
| 3768892a17 | |||
| c804c59f05 | |||
| b219bf829f | |||
| f97b1a6484 | |||
| 0d2cdef10f | |||
| 84db5fe37a | |||
| 128c8e0d08 | |||
| cf4e723f64 | |||
| a3156a156f | |||
| 63a78da18d | |||
| ac50c06cd7 | |||
| b60649f59d | |||
| 6acc9416c1 | |||
| bb4e5d6e3e | |||
| 170c221957 | |||
| 812327f148 | |||
| cd192128f1 | |||
| a5d4d6c11d | |||
| 1452f8d083 | |||
| 33c6706181 | |||
| c9618e1095 | |||
| cae7f3ef63 | |||
| 42793d94bd | |||
| 1369bf0e36 | |||
| 361d14bd3e | |||
| 7923352535 | |||
| 010240066a | |||
| 88b6e2f294 | |||
| 36b0f97897 |
+18
-4
@@ -27,10 +27,24 @@ workflow:
|
||||
.ecr_login: &ecr_login |
|
||||
AWS_CLI_ENV_ARGS=""
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_REGION=$AWS_REGION"
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_ACCESS_KEY_ID=${AWS_ACCESS_KEY_ID:-}"
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_SECRET_ACCESS_KEY=${AWS_SECRET_ACCESS_KEY:-}"
|
||||
if [ -n "${AWS_SESSION_TOKEN:-}" ]; then
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_SESSION_TOKEN=$AWS_SESSION_TOKEN"
|
||||
|
||||
HAS_ACCESS_KEY="false"
|
||||
HAS_SECRET_KEY="false"
|
||||
if [ -n "${AWS_ACCESS_KEY_ID:-}" ]; then
|
||||
HAS_ACCESS_KEY="true"
|
||||
fi
|
||||
if [ -n "${AWS_SECRET_ACCESS_KEY:-}" ]; then
|
||||
HAS_SECRET_KEY="true"
|
||||
fi
|
||||
|
||||
if [ "$HAS_ACCESS_KEY" = "true" ] && [ "$HAS_SECRET_KEY" = "true" ]; then
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_ACCESS_KEY_ID=$AWS_ACCESS_KEY_ID"
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_SECRET_ACCESS_KEY=$AWS_SECRET_ACCESS_KEY"
|
||||
if [ -n "${AWS_SESSION_TOKEN:-}" ]; then
|
||||
AWS_CLI_ENV_ARGS="$AWS_CLI_ENV_ARGS -e AWS_SESSION_TOKEN=$AWS_SESSION_TOKEN"
|
||||
fi
|
||||
elif [ "$HAS_ACCESS_KEY" = "true" ] || [ "$HAS_SECRET_KEY" = "true" ] || [ -n "${AWS_SESSION_TOKEN:-}" ]; then
|
||||
echo "WARN: Incomplete AWS_* env vars detected; ignoring injected AWS credentials for ECR login."
|
||||
fi
|
||||
|
||||
PASS="$(docker run --rm $AWS_CLI_ENV_ARGS public.ecr.aws/aws-cli/aws-cli:latest \
|
||||
|
||||
@@ -0,0 +1,387 @@
|
||||
// Command: fix-stock-log-drift
|
||||
//
|
||||
// Tujuan:
|
||||
// Sinkronkan `stock_logs.stock` (running ledger) dengan `product_warehouses.qty`
|
||||
// (FIFO truth) ketika keduanya drift.
|
||||
//
|
||||
// Drift biasanya terjadi karena bug di Recording-Edit (sebelum fix) yang
|
||||
// hanya menulis -decrease tanpa +increase saat in-place update. Akibatnya
|
||||
// running ledger di stock_logs tertinggal dari qty riil di product_warehouses.
|
||||
//
|
||||
// Cara kerja:
|
||||
// 1. Ambil product_warehouses.qty (sebagai truth)
|
||||
// 2. Ambil last_stock_log.stock
|
||||
// 3. Cari recording yang berkontribusi pada drift (untuk notes)
|
||||
// 4. Hitung drift = qty - last_stock_log.stock
|
||||
// 5. Jika drift != 0, insert 1 stock_log corrective:
|
||||
// - drift > 0 → increase = drift
|
||||
// - drift < 0 → decrease = |drift|
|
||||
// stock akhir akan sama dengan qty (truth).
|
||||
// Notes otomatis berisi daftar recording IDs yang berkontribusi pada drift.
|
||||
//
|
||||
// Mode:
|
||||
// --apply=false (default) → dry-run, hanya tampilkan rencana
|
||||
// --apply=true → eksekusi insert
|
||||
//
|
||||
// Contoh:
|
||||
// go run cmd/fix-stock-log-drift/main.go --product-warehouse-id=1261
|
||||
// go run cmd/fix-stock-log-drift/main.go --product-warehouse-id=1261 --apply
|
||||
// go run cmd/fix-stock-log-drift/main.go --product-warehouse-id=1261 --apply \
|
||||
// --actor-id=1 --notes="Koreksi manual drift"
|
||||
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"flag"
|
||||
"fmt"
|
||||
"log"
|
||||
"math"
|
||||
"os"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/config"
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/database"
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
const (
|
||||
qtyEpsilon = 1e-6
|
||||
defaultActorID uint = 1
|
||||
maxSuspectInNotes = 30
|
||||
)
|
||||
|
||||
type driftRow struct {
|
||||
ProductWarehouseID uint `gorm:"column:product_warehouse_id"`
|
||||
ProductID uint `gorm:"column:product_id"`
|
||||
ProductName string `gorm:"column:product_name"`
|
||||
WarehouseName string `gorm:"column:warehouse_name"`
|
||||
CurrentQty float64 `gorm:"column:current_qty"`
|
||||
LastLogStock float64 `gorm:"column:last_log_stock"`
|
||||
LastLogID uint `gorm:"column:last_log_id"`
|
||||
FifoExpected float64 `gorm:"column:fifo_expected"`
|
||||
}
|
||||
|
||||
type suspectRecording struct {
|
||||
RecordingID uint `gorm:"column:recording_id"`
|
||||
FifoUsage float64 `gorm:"column:fifo_usage"`
|
||||
NetLogConsumed float64 `gorm:"column:net_log_consumed"`
|
||||
Phantom float64 `gorm:"column:phantom"`
|
||||
}
|
||||
|
||||
func main() {
|
||||
var (
|
||||
productWarehouseID uint
|
||||
apply bool
|
||||
actorID uint
|
||||
notes string
|
||||
)
|
||||
|
||||
flag.UintVar(&productWarehouseID, "product-warehouse-id", 0, "Target product_warehouse_id (required)")
|
||||
flag.BoolVar(&apply, "apply", false, "Apply changes. If false, run as dry-run")
|
||||
flag.UintVar(&actorID, "actor-id", defaultActorID, "User id yang akan dicatat sebagai created_by stock_log corrective")
|
||||
flag.StringVar(¬es, "notes", "", "Custom notes untuk stock_log corrective (opsional, default=auto-generate dari data recording)")
|
||||
flag.Parse()
|
||||
|
||||
notes = strings.TrimSpace(notes)
|
||||
if err := validateFlags(productWarehouseID, actorID); err != nil {
|
||||
log.Fatalf("invalid flags: %v", err)
|
||||
}
|
||||
|
||||
ctx := context.Background()
|
||||
db := database.Connect(config.DBHost, config.DBName)
|
||||
|
||||
row, err := loadDriftRow(ctx, db, productWarehouseID)
|
||||
if err != nil {
|
||||
log.Fatalf("failed to load product warehouse: %v", err)
|
||||
}
|
||||
|
||||
suspects, err := loadSuspectRecordings(ctx, db, productWarehouseID)
|
||||
if err != nil {
|
||||
log.Fatalf("failed to load suspect recordings: %v", err)
|
||||
}
|
||||
|
||||
drift := row.CurrentQty - row.LastLogStock
|
||||
|
||||
// Print info header
|
||||
fmt.Printf("Mode: %s\n", modeLabel(apply))
|
||||
fmt.Printf("Target product_warehouse_id: %d\n", productWarehouseID)
|
||||
fmt.Printf("Product: %q\n", row.ProductName)
|
||||
fmt.Printf("Warehouse: %q\n", row.WarehouseName)
|
||||
fmt.Printf("Current qty (product_warehouses): %.3f\n", row.CurrentQty)
|
||||
fmt.Printf("FIFO expected (sum total_qty - total_used): %.3f\n", row.FifoExpected)
|
||||
fmt.Printf("Last stock_log: id=%d stock=%.3f\n", row.LastLogID, row.LastLogStock)
|
||||
fmt.Printf("Drift (qty - last_log_stock): %+.3f\n", drift)
|
||||
|
||||
if !nearlyEqual(row.CurrentQty, row.FifoExpected) {
|
||||
fmt.Println()
|
||||
fmt.Println("⚠️ WARNING: product_warehouses.qty TIDAK match dengan FIFO expected.")
|
||||
fmt.Println(" Disarankan jalankan dulu cmd/reflow-quantity-product-warehouse-from-stock-allocation")
|
||||
fmt.Println(" sebelum fix stock_log drift, agar truth source-nya sudah benar.")
|
||||
}
|
||||
|
||||
// Print suspect recordings
|
||||
fmt.Println()
|
||||
if len(suspects) > 0 {
|
||||
totalPhantom := 0.0
|
||||
for _, s := range suspects {
|
||||
totalPhantom += s.Phantom
|
||||
}
|
||||
fmt.Printf("Suspect recordings (drift contributors): %d\n", len(suspects))
|
||||
for _, s := range suspects {
|
||||
fmt.Printf(
|
||||
" #%-6d fifo=%-10.3f net_log=%-10.3f phantom=%+.3f\n",
|
||||
s.RecordingID, s.FifoUsage, s.NetLogConsumed, s.Phantom,
|
||||
)
|
||||
}
|
||||
fmt.Printf("Total suspect phantom: %+.3f\n", totalPhantom)
|
||||
} else {
|
||||
fmt.Println("Suspect recordings: none found (drift origin unknown)")
|
||||
}
|
||||
fmt.Println()
|
||||
|
||||
if nearlyEqual(drift, 0) {
|
||||
fmt.Println("✓ Tidak ada drift. Stock_log sudah sinkron dengan product_warehouses.qty.")
|
||||
fmt.Println("Summary: planned=0 inserted=0 skipped=1 failed=0")
|
||||
return
|
||||
}
|
||||
|
||||
// Build notes if not provided
|
||||
if notes == "" {
|
||||
notes = buildDefaultNotes(row, drift, suspects)
|
||||
}
|
||||
|
||||
plan := buildCorrectiveLog(row, drift, actorID, notes)
|
||||
|
||||
fmt.Printf(
|
||||
"PLAN insert stock_log:\n pw=%d increase=%.3f decrease=%.3f stock=%.3f\n notes=%q\n",
|
||||
plan.ProductWarehouseId,
|
||||
plan.Increase,
|
||||
plan.Decrease,
|
||||
plan.Stock,
|
||||
plan.Notes,
|
||||
)
|
||||
|
||||
if !apply {
|
||||
fmt.Println()
|
||||
fmt.Println("Summary: planned=1 inserted=0 skipped=0 failed=0 (dry-run)")
|
||||
return
|
||||
}
|
||||
|
||||
if err := db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
|
||||
// Re-check di dalam transaction agar aman dari race condition
|
||||
current, err := loadDriftRow(ctx, tx, productWarehouseID)
|
||||
if err != nil {
|
||||
return fmt.Errorf("re-read product_warehouse_id=%d: %w", productWarehouseID, err)
|
||||
}
|
||||
currentDrift := current.CurrentQty - current.LastLogStock
|
||||
if nearlyEqual(currentDrift, 0) {
|
||||
fmt.Println("Drift hilang sebelum insert (kemungkinan ada operasi paralel). Skip.")
|
||||
return nil
|
||||
}
|
||||
fresh := buildCorrectiveLog(current, currentDrift, actorID, notes)
|
||||
if err := tx.Create(&fresh).Error; err != nil {
|
||||
return fmt.Errorf("insert corrective stock_log for pw=%d: %w", productWarehouseID, err)
|
||||
}
|
||||
fmt.Printf(
|
||||
"DONE inserted stock_log id=%d pw=%d increase=%.3f decrease=%.3f stock=%.3f\n",
|
||||
fresh.Id,
|
||||
fresh.ProductWarehouseId,
|
||||
fresh.Increase,
|
||||
fresh.Decrease,
|
||||
fresh.Stock,
|
||||
)
|
||||
return nil
|
||||
}); err != nil {
|
||||
fmt.Println()
|
||||
fmt.Println("Summary: planned=1 inserted=0 skipped=0 failed=1")
|
||||
log.Printf("error: %v", err)
|
||||
os.Exit(1)
|
||||
}
|
||||
|
||||
fmt.Println()
|
||||
fmt.Println("Summary: planned=1 inserted=1 skipped=0 failed=0")
|
||||
}
|
||||
|
||||
func validateFlags(productWarehouseID uint, actorID uint) error {
|
||||
if productWarehouseID == 0 {
|
||||
return errors.New("--product-warehouse-id is required (must be > 0)")
|
||||
}
|
||||
if actorID == 0 {
|
||||
return errors.New("--actor-id must be > 0")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func loadDriftRow(ctx context.Context, db *gorm.DB, productWarehouseID uint) (driftRow, error) {
|
||||
row := driftRow{}
|
||||
|
||||
lastLogSub := db.WithContext(ctx).
|
||||
Table("stock_logs").
|
||||
Select("id, product_warehouse_id, stock").
|
||||
Where("product_warehouse_id = ?", productWarehouseID).
|
||||
Order("id DESC").
|
||||
Limit(1)
|
||||
|
||||
fifoSub := db.WithContext(ctx).
|
||||
Table("purchase_items").
|
||||
Select(`
|
||||
product_warehouse_id,
|
||||
COALESCE(SUM(COALESCE(total_qty, 0) - COALESCE(total_used, 0)), 0) AS fifo_expected
|
||||
`).
|
||||
Where("product_warehouse_id = ?", productWarehouseID).
|
||||
Group("product_warehouse_id")
|
||||
|
||||
if err := db.WithContext(ctx).
|
||||
Table("product_warehouses pw").
|
||||
Select(`
|
||||
pw.id AS product_warehouse_id,
|
||||
pw.product_id AS product_id,
|
||||
COALESCE(p.name, '') AS product_name,
|
||||
COALESCE(w.name, '') AS warehouse_name,
|
||||
COALESCE(pw.qty, 0) AS current_qty,
|
||||
COALESCE(last_log.stock, 0) AS last_log_stock,
|
||||
COALESCE(last_log.id, 0) AS last_log_id,
|
||||
COALESCE(fifo.fifo_expected, 0) AS fifo_expected
|
||||
`).
|
||||
Joins("LEFT JOIN products p ON p.id = pw.product_id").
|
||||
Joins("LEFT JOIN warehouses w ON w.id = pw.warehouse_id").
|
||||
Joins("LEFT JOIN (?) last_log ON last_log.product_warehouse_id = pw.id", lastLogSub).
|
||||
Joins("LEFT JOIN (?) fifo ON fifo.product_warehouse_id = pw.id", fifoSub).
|
||||
Where("pw.id = ?", productWarehouseID).
|
||||
Scan(&row).Error; err != nil {
|
||||
return row, err
|
||||
}
|
||||
|
||||
if row.ProductWarehouseID == 0 {
|
||||
return row, fmt.Errorf("product_warehouse_id=%d not found", productWarehouseID)
|
||||
}
|
||||
|
||||
return row, nil
|
||||
}
|
||||
|
||||
// loadSuspectRecordings mencari recording yang net stock_log consumed-nya
|
||||
// melebihi FIFO usage_qty — ini adalah recording yang berkontribusi pada drift
|
||||
// akibat bug Recording-Edit yang hanya menulis -decrease tanpa +increase.
|
||||
func loadSuspectRecordings(ctx context.Context, db *gorm.DB, productWarehouseID uint) ([]suspectRecording, error) {
|
||||
rows := make([]suspectRecording, 0)
|
||||
if err := db.WithContext(ctx).
|
||||
Table("recording_stocks rs").
|
||||
Select(`
|
||||
rs.recording_id,
|
||||
COALESCE(rs.usage_qty, 0) AS fifo_usage,
|
||||
(
|
||||
COALESCE(SUM(CASE WHEN sl.decrease > 0 THEN sl.decrease ELSE 0 END), 0) -
|
||||
COALESCE(SUM(CASE WHEN sl.increase > 0 THEN sl.increase ELSE 0 END), 0)
|
||||
) AS net_log_consumed,
|
||||
(
|
||||
COALESCE(SUM(CASE WHEN sl.decrease > 0 THEN sl.decrease ELSE 0 END), 0) -
|
||||
COALESCE(SUM(CASE WHEN sl.increase > 0 THEN sl.increase ELSE 0 END), 0) -
|
||||
COALESCE(rs.usage_qty, 0)
|
||||
) AS phantom
|
||||
`).
|
||||
Joins(`
|
||||
JOIN stock_logs sl ON sl.loggable_type = ?
|
||||
AND sl.loggable_id = rs.recording_id
|
||||
AND sl.product_warehouse_id = rs.product_warehouse_id
|
||||
`, string(utils.StockLogTypeRecording)).
|
||||
Where("rs.product_warehouse_id = ?", productWarehouseID).
|
||||
Group("rs.recording_id, rs.usage_qty").
|
||||
Having(`
|
||||
ABS(
|
||||
COALESCE(SUM(CASE WHEN sl.decrease > 0 THEN sl.decrease ELSE 0 END), 0) -
|
||||
COALESCE(SUM(CASE WHEN sl.increase > 0 THEN sl.increase ELSE 0 END), 0) -
|
||||
COALESCE(rs.usage_qty, 0)
|
||||
) > ?
|
||||
`, qtyEpsilon).
|
||||
Order("rs.recording_id ASC").
|
||||
Scan(&rows).Error; err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return rows, nil
|
||||
}
|
||||
|
||||
// buildDefaultNotes membuat notes otomatis yang berisi penjelasan drift
|
||||
// beserta daftar recording_id yang berkontribusi + phantom amount masing-masing.
|
||||
func buildDefaultNotes(row driftRow, drift float64, suspects []suspectRecording) string {
|
||||
sign := "+"
|
||||
if drift < 0 {
|
||||
sign = ""
|
||||
}
|
||||
|
||||
var sb strings.Builder
|
||||
sb.WriteString(fmt.Sprintf(
|
||||
"Koreksi drift stock_log akibat bug Recording-Edit (in-place update menulis -decrease tanpa +increase). PW=%d (%s) drift=%s%.3f.",
|
||||
row.ProductWarehouseID,
|
||||
row.WarehouseName,
|
||||
sign,
|
||||
drift,
|
||||
))
|
||||
|
||||
if len(suspects) == 0 {
|
||||
return sb.String()
|
||||
}
|
||||
|
||||
sb.WriteString(" Recordings affected:")
|
||||
|
||||
limit := len(suspects)
|
||||
truncated := 0
|
||||
if limit > maxSuspectInNotes {
|
||||
truncated = limit - maxSuspectInNotes
|
||||
limit = maxSuspectInNotes
|
||||
}
|
||||
|
||||
for i := 0; i < limit; i++ {
|
||||
s := suspects[i]
|
||||
phantomSign := "+"
|
||||
if s.Phantom < 0 {
|
||||
phantomSign = ""
|
||||
}
|
||||
sb.WriteString(fmt.Sprintf(" #%d(%s%.0f)", s.RecordingID, phantomSign, s.Phantom))
|
||||
if i < limit-1 || truncated > 0 {
|
||||
sb.WriteString(",")
|
||||
}
|
||||
}
|
||||
|
||||
if truncated > 0 {
|
||||
sb.WriteString(fmt.Sprintf(" ... (+%d more)", truncated))
|
||||
}
|
||||
|
||||
sb.WriteString(".")
|
||||
return sb.String()
|
||||
}
|
||||
|
||||
func buildCorrectiveLog(row driftRow, drift float64, actorID uint, notes string) entity.StockLog {
|
||||
corrective := entity.StockLog{
|
||||
ProductWarehouseId: row.ProductWarehouseID,
|
||||
CreatedBy: actorID,
|
||||
LoggableType: string(utils.StockLogTypeAdjustment),
|
||||
LoggableId: 0,
|
||||
Stock: row.CurrentQty,
|
||||
Notes: notes,
|
||||
CreatedAt: time.Now(),
|
||||
}
|
||||
if drift > 0 {
|
||||
corrective.Increase = drift
|
||||
corrective.Decrease = 0
|
||||
} else {
|
||||
corrective.Increase = 0
|
||||
corrective.Decrease = -drift
|
||||
}
|
||||
return corrective
|
||||
}
|
||||
|
||||
func modeLabel(apply bool) string {
|
||||
if apply {
|
||||
return "APPLY"
|
||||
}
|
||||
return "DRY-RUN"
|
||||
}
|
||||
|
||||
func nearlyEqual(a, b float64) bool {
|
||||
return math.Abs(a-b) <= qtyEpsilon
|
||||
}
|
||||
@@ -3215,6 +3215,55 @@
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/inventory/stock-logs/": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/inventory/stock-logs`.",
|
||||
"responses": {
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Successful response"
|
||||
},
|
||||
"401": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Unauthorized"
|
||||
},
|
||||
"403": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Forbidden"
|
||||
}
|
||||
},
|
||||
"security": [
|
||||
{
|
||||
"ApiKeyAuth": []
|
||||
},
|
||||
{
|
||||
"BearerAuth": []
|
||||
}
|
||||
],
|
||||
"summary": "GET api / inventory / stock logs",
|
||||
"tags": [
|
||||
"Inventory"
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/inventory/transfers/": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/inventory/transfers`.",
|
||||
@@ -4318,6 +4367,29 @@
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"example": {
|
||||
"code": 200,
|
||||
"data": [
|
||||
{
|
||||
"created_at": "2026-01-01T00:00:00Z",
|
||||
"created_user": {
|
||||
"id": 1,
|
||||
"name": "Admin"
|
||||
},
|
||||
"id": 1,
|
||||
"name": "FCR Broiler Standard",
|
||||
"updated_at": "2026-01-01T00:00:00Z"
|
||||
}
|
||||
],
|
||||
"message": "Get all fcrs successfully",
|
||||
"meta": {
|
||||
"limit": 10,
|
||||
"page": 1,
|
||||
"total_pages": 1,
|
||||
"total_results": 1
|
||||
},
|
||||
"status": "success"
|
||||
},
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
@@ -4379,6 +4451,41 @@
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"example": {
|
||||
"code": 200,
|
||||
"data": {
|
||||
"created_at": "2026-01-01T00:00:00Z",
|
||||
"created_user": {
|
||||
"id": 1,
|
||||
"name": "Admin"
|
||||
},
|
||||
"fcr_standards": [
|
||||
{
|
||||
"fcr_number": 1.2,
|
||||
"id": 1,
|
||||
"mortality": 0.5,
|
||||
"weight": 0.5
|
||||
},
|
||||
{
|
||||
"fcr_number": 1.35,
|
||||
"id": 2,
|
||||
"mortality": 0.3,
|
||||
"weight": 1
|
||||
},
|
||||
{
|
||||
"fcr_number": 1.5,
|
||||
"id": 3,
|
||||
"mortality": 0.25,
|
||||
"weight": 1.5
|
||||
}
|
||||
],
|
||||
"id": 1,
|
||||
"name": "FCR Broiler Standard",
|
||||
"updated_at": "2026-01-01T00:00:00Z"
|
||||
},
|
||||
"message": "Get fcr successfully",
|
||||
"status": "success"
|
||||
},
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/SuccessEnvelope"
|
||||
}
|
||||
@@ -6457,6 +6564,126 @@
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/production/chickins/": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/production/chickins`.",
|
||||
"parameters": [
|
||||
{
|
||||
"description": "Page number.",
|
||||
"example": 1,
|
||||
"in": "query",
|
||||
"name": "page",
|
||||
"required": false,
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
}
|
||||
},
|
||||
{
|
||||
"description": "Page size.",
|
||||
"example": 10,
|
||||
"in": "query",
|
||||
"name": "limit",
|
||||
"required": false,
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
}
|
||||
},
|
||||
{
|
||||
"description": "Project flock kandang id filter.",
|
||||
"example": 1,
|
||||
"in": "query",
|
||||
"name": "project_flock_kandang_id",
|
||||
"required": false,
|
||||
"schema": {
|
||||
"type": "integer"
|
||||
}
|
||||
}
|
||||
],
|
||||
"responses": {
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"example": {
|
||||
"code": 200,
|
||||
"data": [
|
||||
{
|
||||
"chick_in_date": "2026-01-01T00:00:00Z",
|
||||
"created_at": "2026-01-01T00:00:00Z",
|
||||
"created_user": {
|
||||
"id": 1,
|
||||
"name": "Admin"
|
||||
},
|
||||
"id": 1,
|
||||
"notes": "",
|
||||
"pending_usage_qty": 0,
|
||||
"product_warehouse": {
|
||||
"id": 1,
|
||||
"product": {
|
||||
"id": 1,
|
||||
"name": "DOC Broiler"
|
||||
},
|
||||
"warehouse": {
|
||||
"id": 1,
|
||||
"name": "Gudang DOC"
|
||||
}
|
||||
},
|
||||
"product_warehouse_id": 1,
|
||||
"project_flock_kandang_id": 1,
|
||||
"updated_at": "2026-01-01T00:00:00Z",
|
||||
"usage_qty": 10000
|
||||
}
|
||||
],
|
||||
"message": "Get all chickins successfully",
|
||||
"meta": {
|
||||
"limit": 10,
|
||||
"page": 1,
|
||||
"total_pages": 1,
|
||||
"total_results": 1
|
||||
},
|
||||
"status": "success"
|
||||
},
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Successful response"
|
||||
},
|
||||
"401": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Unauthorized"
|
||||
},
|
||||
"403": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Forbidden"
|
||||
}
|
||||
},
|
||||
"security": [
|
||||
{
|
||||
"ApiKeyAuth": []
|
||||
},
|
||||
{
|
||||
"BearerAuth": []
|
||||
}
|
||||
],
|
||||
"summary": "GET api / production / chickins",
|
||||
"tags": [
|
||||
"Production"
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/production/chickins/{id}": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/production/chickins/:id`.",
|
||||
@@ -7517,6 +7744,47 @@
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"example": {
|
||||
"code": 200,
|
||||
"data": [
|
||||
{
|
||||
"approval": {
|
||||
"action": null,
|
||||
"step_name": "Pengajuan",
|
||||
"step_number": 1
|
||||
},
|
||||
"created_at": "2026-01-15T00:00:00Z",
|
||||
"created_by": 1,
|
||||
"created_user": {
|
||||
"id": 1,
|
||||
"name": "Admin"
|
||||
},
|
||||
"economic_cutoff_date": "2026-01-20T00:00:00Z",
|
||||
"effective_move_date": "2026-01-18T00:00:00Z",
|
||||
"executed_at": null,
|
||||
"from_project_flock": {
|
||||
"flock_name": "Flock A Period 1",
|
||||
"id": 1
|
||||
},
|
||||
"id": 1,
|
||||
"notes": "",
|
||||
"to_project_flock": {
|
||||
"flock_name": "Flock B Period 1",
|
||||
"id": 2
|
||||
},
|
||||
"transfer_date": "2026-01-15T00:00:00Z",
|
||||
"transfer_number": "TL-00001"
|
||||
}
|
||||
],
|
||||
"message": "Get all transferLayings successfully",
|
||||
"meta": {
|
||||
"limit": 10,
|
||||
"page": 1,
|
||||
"total_pages": 1,
|
||||
"total_results": 1
|
||||
},
|
||||
"status": "success"
|
||||
},
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
@@ -7700,6 +7968,69 @@
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"example": {
|
||||
"code": 200,
|
||||
"data": {
|
||||
"approval": {
|
||||
"action": null,
|
||||
"step_name": "Pengajuan",
|
||||
"step_number": 1
|
||||
},
|
||||
"created_at": "2026-01-15T00:00:00Z",
|
||||
"created_by": 1,
|
||||
"created_user": {
|
||||
"id": 1,
|
||||
"name": "Admin"
|
||||
},
|
||||
"economic_cutoff_date": "2026-01-20T00:00:00Z",
|
||||
"effective_move_date": "2026-01-18T00:00:00Z",
|
||||
"executed_at": null,
|
||||
"from_project_flock": {
|
||||
"flock_name": "Flock A Period 1",
|
||||
"id": 1
|
||||
},
|
||||
"id": 1,
|
||||
"notes": "",
|
||||
"sources": [
|
||||
{
|
||||
"note": "",
|
||||
"qty": 5000,
|
||||
"source_project_flock_kandang": {
|
||||
"id": 1,
|
||||
"kandang": {
|
||||
"id": 1,
|
||||
"name": "Kandang A"
|
||||
},
|
||||
"kandang_id": 1,
|
||||
"project_flock_id": 1
|
||||
}
|
||||
}
|
||||
],
|
||||
"targets": [
|
||||
{
|
||||
"note": "",
|
||||
"qty": 5000,
|
||||
"target_project_flock_kandang": {
|
||||
"id": 2,
|
||||
"kandang": {
|
||||
"id": 2,
|
||||
"name": "Kandang B"
|
||||
},
|
||||
"kandang_id": 2,
|
||||
"project_flock_id": 2
|
||||
}
|
||||
}
|
||||
],
|
||||
"to_project_flock": {
|
||||
"flock_name": "Flock B Period 1",
|
||||
"id": 2
|
||||
},
|
||||
"transfer_date": "2026-01-15T00:00:00Z",
|
||||
"transfer_number": "TL-00001"
|
||||
},
|
||||
"message": "Get transferLaying successfully",
|
||||
"status": "success"
|
||||
},
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/SuccessEnvelope"
|
||||
}
|
||||
@@ -8912,6 +9243,55 @@
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/reports/hpp-v2-breakdown": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/reports/hpp-v2-breakdown`.",
|
||||
"responses": {
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Successful response"
|
||||
},
|
||||
"401": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Unauthorized"
|
||||
},
|
||||
"403": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Forbidden"
|
||||
}
|
||||
},
|
||||
"security": [
|
||||
{
|
||||
"ApiKeyAuth": []
|
||||
},
|
||||
{
|
||||
"BearerAuth": []
|
||||
}
|
||||
],
|
||||
"summary": "GET api / reports / hpp v2 breakdown",
|
||||
"tags": [
|
||||
"Reports"
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/reports/marketing": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/reports/marketing`.",
|
||||
@@ -9555,6 +9935,55 @@
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/system-settings/": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/system-settings`.",
|
||||
"responses": {
|
||||
"200": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/PaginatedEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Successful response"
|
||||
},
|
||||
"401": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Unauthorized"
|
||||
},
|
||||
"403": {
|
||||
"content": {
|
||||
"application/json": {
|
||||
"schema": {
|
||||
"$ref": "#/components/schemas/ErrorEnvelope"
|
||||
}
|
||||
}
|
||||
},
|
||||
"description": "Forbidden"
|
||||
}
|
||||
},
|
||||
"security": [
|
||||
{
|
||||
"ApiKeyAuth": []
|
||||
},
|
||||
{
|
||||
"BearerAuth": []
|
||||
}
|
||||
],
|
||||
"summary": "GET api / system settings",
|
||||
"tags": [
|
||||
"API"
|
||||
]
|
||||
}
|
||||
},
|
||||
"/api/users/": {
|
||||
"get": {
|
||||
"description": "Read access to `/api/users`.",
|
||||
|
||||
@@ -2006,6 +2006,34 @@ paths:
|
||||
summary: GET api / inventory / product warehouses / :id
|
||||
tags:
|
||||
- Inventory
|
||||
/api/inventory/stock-logs/:
|
||||
get:
|
||||
description: Read access to `/api/inventory/stock-logs`.
|
||||
responses:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
"401":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Unauthorized
|
||||
"403":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Forbidden
|
||||
security:
|
||||
- ApiKeyAuth: []
|
||||
- BearerAuth: []
|
||||
summary: GET api / inventory / stock logs
|
||||
tags:
|
||||
- Inventory
|
||||
/api/inventory/transfers/:
|
||||
get:
|
||||
description: Read access to `/api/inventory/transfers`.
|
||||
@@ -2686,6 +2714,23 @@ paths:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
example:
|
||||
code: 200
|
||||
data:
|
||||
- created_at: "2026-01-01T00:00:00Z"
|
||||
created_user:
|
||||
id: 1
|
||||
name: Admin
|
||||
id: 1
|
||||
name: FCR Broiler Standard
|
||||
updated_at: "2026-01-01T00:00:00Z"
|
||||
message: Get all fcrs successfully
|
||||
meta:
|
||||
limit: 10
|
||||
page: 1
|
||||
total_pages: 1
|
||||
total_results: 1
|
||||
status: success
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
@@ -2722,6 +2767,31 @@ paths:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
example:
|
||||
code: 200
|
||||
data:
|
||||
created_at: "2026-01-01T00:00:00Z"
|
||||
created_user:
|
||||
id: 1
|
||||
name: Admin
|
||||
fcr_standards:
|
||||
- fcr_number: 1.2
|
||||
id: 1
|
||||
mortality: 0.5
|
||||
weight: 0.5
|
||||
- fcr_number: 1.35
|
||||
id: 2
|
||||
mortality: 0.3
|
||||
weight: 1
|
||||
- fcr_number: 1.5
|
||||
id: 3
|
||||
mortality: 0.25
|
||||
weight: 1.5
|
||||
id: 1
|
||||
name: FCR Broiler Standard
|
||||
updated_at: "2026-01-01T00:00:00Z"
|
||||
message: Get fcr successfully
|
||||
status: success
|
||||
schema:
|
||||
$ref: '#/components/schemas/SuccessEnvelope'
|
||||
description: Successful response
|
||||
@@ -3994,6 +4064,86 @@ paths:
|
||||
summary: GET api / master data / warehouses / :id
|
||||
tags:
|
||||
- Master Data
|
||||
/api/production/chickins/:
|
||||
get:
|
||||
description: Read access to `/api/production/chickins`.
|
||||
parameters:
|
||||
- description: Page number.
|
||||
example: 1
|
||||
in: query
|
||||
name: page
|
||||
required: false
|
||||
schema:
|
||||
type: integer
|
||||
- description: Page size.
|
||||
example: 10
|
||||
in: query
|
||||
name: limit
|
||||
required: false
|
||||
schema:
|
||||
type: integer
|
||||
- description: Project flock kandang id filter.
|
||||
example: 1
|
||||
in: query
|
||||
name: project_flock_kandang_id
|
||||
required: false
|
||||
schema:
|
||||
type: integer
|
||||
responses:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
example:
|
||||
code: 200
|
||||
data:
|
||||
- chick_in_date: "2026-01-01T00:00:00Z"
|
||||
created_at: "2026-01-01T00:00:00Z"
|
||||
created_user:
|
||||
id: 1
|
||||
name: Admin
|
||||
id: 1
|
||||
notes: ""
|
||||
pending_usage_qty: 0
|
||||
product_warehouse:
|
||||
id: 1
|
||||
product:
|
||||
id: 1
|
||||
name: DOC Broiler
|
||||
warehouse:
|
||||
id: 1
|
||||
name: Gudang DOC
|
||||
product_warehouse_id: 1
|
||||
project_flock_kandang_id: 1
|
||||
updated_at: "2026-01-01T00:00:00Z"
|
||||
usage_qty: 10000
|
||||
message: Get all chickins successfully
|
||||
meta:
|
||||
limit: 10
|
||||
page: 1
|
||||
total_pages: 1
|
||||
total_results: 1
|
||||
status: success
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
"401":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Unauthorized
|
||||
"403":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Forbidden
|
||||
security:
|
||||
- ApiKeyAuth: []
|
||||
- BearerAuth: []
|
||||
summary: GET api / production / chickins
|
||||
tags:
|
||||
- Production
|
||||
/api/production/chickins/{id}:
|
||||
get:
|
||||
description: Read access to `/api/production/chickins/:id`.
|
||||
@@ -4664,6 +4814,38 @@ paths:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
example:
|
||||
code: 200
|
||||
data:
|
||||
- approval:
|
||||
action: null
|
||||
step_name: Pengajuan
|
||||
step_number: 1
|
||||
created_at: "2026-01-15T00:00:00Z"
|
||||
created_by: 1
|
||||
created_user:
|
||||
id: 1
|
||||
name: Admin
|
||||
economic_cutoff_date: "2026-01-20T00:00:00Z"
|
||||
effective_move_date: "2026-01-18T00:00:00Z"
|
||||
executed_at: null
|
||||
from_project_flock:
|
||||
flock_name: Flock A Period 1
|
||||
id: 1
|
||||
id: 1
|
||||
notes: ""
|
||||
to_project_flock:
|
||||
flock_name: Flock B Period 1
|
||||
id: 2
|
||||
transfer_date: "2026-01-15T00:00:00Z"
|
||||
transfer_number: TL-00001
|
||||
message: Get all transferLayings successfully
|
||||
meta:
|
||||
limit: 10
|
||||
page: 1
|
||||
total_pages: 1
|
||||
total_results: 1
|
||||
status: success
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
@@ -4700,6 +4882,53 @@ paths:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
example:
|
||||
code: 200
|
||||
data:
|
||||
approval:
|
||||
action: null
|
||||
step_name: Pengajuan
|
||||
step_number: 1
|
||||
created_at: "2026-01-15T00:00:00Z"
|
||||
created_by: 1
|
||||
created_user:
|
||||
id: 1
|
||||
name: Admin
|
||||
economic_cutoff_date: "2026-01-20T00:00:00Z"
|
||||
effective_move_date: "2026-01-18T00:00:00Z"
|
||||
executed_at: null
|
||||
from_project_flock:
|
||||
flock_name: Flock A Period 1
|
||||
id: 1
|
||||
id: 1
|
||||
notes: ""
|
||||
sources:
|
||||
- note: ""
|
||||
qty: 5000
|
||||
source_project_flock_kandang:
|
||||
id: 1
|
||||
kandang:
|
||||
id: 1
|
||||
name: Kandang A
|
||||
kandang_id: 1
|
||||
project_flock_id: 1
|
||||
targets:
|
||||
- note: ""
|
||||
qty: 5000
|
||||
target_project_flock_kandang:
|
||||
id: 2
|
||||
kandang:
|
||||
id: 2
|
||||
name: Kandang B
|
||||
kandang_id: 2
|
||||
project_flock_id: 2
|
||||
to_project_flock:
|
||||
flock_name: Flock B Period 1
|
||||
id: 2
|
||||
transfer_date: "2026-01-15T00:00:00Z"
|
||||
transfer_number: TL-00001
|
||||
message: Get transferLaying successfully
|
||||
status: success
|
||||
schema:
|
||||
$ref: '#/components/schemas/SuccessEnvelope'
|
||||
description: Successful response
|
||||
@@ -5545,6 +5774,34 @@ paths:
|
||||
summary: GET api / reports / hpp per kandang
|
||||
tags:
|
||||
- Reports
|
||||
/api/reports/hpp-v2-breakdown:
|
||||
get:
|
||||
description: Read access to `/api/reports/hpp-v2-breakdown`.
|
||||
responses:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
"401":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Unauthorized
|
||||
"403":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Forbidden
|
||||
security:
|
||||
- ApiKeyAuth: []
|
||||
- BearerAuth: []
|
||||
summary: GET api / reports / hpp v2 breakdown
|
||||
tags:
|
||||
- Reports
|
||||
/api/reports/marketing:
|
||||
get:
|
||||
description: Read access to `/api/reports/marketing`.
|
||||
@@ -5955,6 +6212,34 @@ paths:
|
||||
summary: GET api / sso / userinfo
|
||||
tags:
|
||||
- SSO
|
||||
/api/system-settings/:
|
||||
get:
|
||||
description: Read access to `/api/system-settings`.
|
||||
responses:
|
||||
"200":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/PaginatedEnvelope'
|
||||
description: Successful response
|
||||
"401":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Unauthorized
|
||||
"403":
|
||||
content:
|
||||
application/json:
|
||||
schema:
|
||||
$ref: '#/components/schemas/ErrorEnvelope'
|
||||
description: Forbidden
|
||||
security:
|
||||
- ApiKeyAuth: []
|
||||
- BearerAuth: []
|
||||
summary: GET api / system settings
|
||||
tags:
|
||||
- API
|
||||
/api/users/:
|
||||
get:
|
||||
description: Read access to `/api/users`.
|
||||
|
||||
@@ -109,6 +109,19 @@
|
||||
"method": "GET",
|
||||
"url": "{{base_url}}/api/closings/?page=1\u0026limit=10\u0026search=kandang\u0026project_status=1\u0026location_id={{location_id}}"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / system settings",
|
||||
"request": {
|
||||
"header": [
|
||||
{
|
||||
"key": "Accept",
|
||||
"value": "application/json"
|
||||
}
|
||||
],
|
||||
"method": "GET",
|
||||
"url": "{{base_url}}/api/system-settings/"
|
||||
}
|
||||
}
|
||||
],
|
||||
"name": "API"
|
||||
@@ -582,6 +595,19 @@
|
||||
"url": "{{base_url}}/api/inventory/product-warehouses/{{id}}"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / inventory / stock logs",
|
||||
"request": {
|
||||
"header": [
|
||||
{
|
||||
"key": "Accept",
|
||||
"value": "application/json"
|
||||
}
|
||||
],
|
||||
"method": "GET",
|
||||
"url": "{{base_url}}/api/inventory/stock-logs/"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / inventory / transfers",
|
||||
"request": {
|
||||
@@ -1143,6 +1169,19 @@
|
||||
},
|
||||
{
|
||||
"item": [
|
||||
{
|
||||
"name": "GET api / production / chickins",
|
||||
"request": {
|
||||
"header": [
|
||||
{
|
||||
"key": "Accept",
|
||||
"value": "application/json"
|
||||
}
|
||||
],
|
||||
"method": "GET",
|
||||
"url": "{{base_url}}/api/production/chickins/?page=1\u0026limit=10\u0026project_flock_kandang_id={{project_flock_kandang_id}}"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / production / chickins / :id",
|
||||
"request": {
|
||||
@@ -1478,6 +1517,19 @@
|
||||
"url": "{{base_url}}/api/reports/hpp-per-kandang?page=1\u0026limit=10\u0026period=2026-01-01\u0026show_unrecorded=false\u0026area_id=1,2\u0026location_id=1,2\u0026kandang_id=1,2\u0026weight_min=1.2\u0026weight_max=1.8"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / reports / hpp v2 breakdown",
|
||||
"request": {
|
||||
"header": [
|
||||
{
|
||||
"key": "Accept",
|
||||
"value": "application/json"
|
||||
}
|
||||
],
|
||||
"method": "GET",
|
||||
"url": "{{base_url}}/api/reports/hpp-v2-breakdown"
|
||||
}
|
||||
},
|
||||
{
|
||||
"name": "GET api / reports / marketing",
|
||||
"request": {
|
||||
|
||||
@@ -89,5 +89,6 @@ func DefaultDashboardPermissions() []string {
|
||||
"lti.users.detail",
|
||||
"lti.users.list",
|
||||
"lti.daily_checklist.master_data.kandang",
|
||||
"lti.production.chickins.list",
|
||||
}
|
||||
}
|
||||
|
||||
@@ -102,11 +102,17 @@ type HppV2CostRepository interface {
|
||||
GetProjectFlockKandangContext(ctx context.Context, projectFlockKandangId uint) (*HppV2ProjectFlockKandangContext, error)
|
||||
GetProjectFlockKandangIDs(ctx context.Context, projectFlockId uint) ([]uint, error)
|
||||
GetLatestTransferInputByProjectFlockKandangID(ctx context.Context, projectFlockKandangId uint, period time.Time) (*HppV2LatestTransferInputRow, error)
|
||||
// GetAllTransferInputsByProjectFlockKandangID return SEMUA approved transfer ke target kandang
|
||||
// itu, untuk skenario multi-source di mana 1 target menerima dari multiple transfer terpisah.
|
||||
// Setiap row = 1 transfer dengan cost basis & chick_in_date sendiri (per source). Order:
|
||||
// effective_date ASC, id ASC (kronologis).
|
||||
GetAllTransferInputsByProjectFlockKandangID(ctx context.Context, projectFlockKandangId uint, period time.Time) ([]HppV2LatestTransferInputRow, error)
|
||||
GetManualDepreciationInputByProjectFlockID(ctx context.Context, projectFlockID uint) (*HppV2ManualDepreciationInputRow, error)
|
||||
GetRecordingStockRoutingAdjustmentCostByProjectFlockID(ctx context.Context, projectFlockID uint, periodDate time.Time) (float64, error)
|
||||
GetFarmDepreciationSnapshotByProjectFlockIDAndPeriod(ctx context.Context, projectFlockID uint, periodDate time.Time) (*HppV2FarmDepreciationSnapshotRow, error)
|
||||
GetEarliestChickInDateByProjectFlockID(ctx context.Context, projectFlockID uint) (*time.Time, error)
|
||||
GetDepreciationPercents(ctx context.Context, houseTypes []string, maxDay int) (map[string]map[int]float64, error)
|
||||
GetChickinPopulationByPFKForFarm(ctx context.Context, projectFlockID uint) (map[uint]float64, error)
|
||||
GetMultiplicationPercentages(ctx context.Context, houseTypes []string, maxDay int) (map[string]map[int]float64, error)
|
||||
ListUsageCostRowsByProductFlags(ctx context.Context, projectFlockKandangIDs []uint, flagNames []string, date *time.Time) ([]HppV2UsageCostRow, 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)
|
||||
@@ -230,6 +236,62 @@ LIMIT 1
|
||||
return &row, nil
|
||||
}
|
||||
|
||||
func (r *HppV2RepositoryImpl) GetAllTransferInputsByProjectFlockKandangID(
|
||||
ctx context.Context,
|
||||
projectFlockKandangId uint,
|
||||
period time.Time,
|
||||
) ([]HppV2LatestTransferInputRow, error) {
|
||||
var rows []HppV2LatestTransferInputRow
|
||||
query := `
|
||||
WITH latest_transfer_approval AS (
|
||||
SELECT a.approvable_id, a.action
|
||||
FROM approvals a
|
||||
JOIN (
|
||||
SELECT approvable_id, MAX(action_at) AS latest_action_at
|
||||
FROM approvals
|
||||
WHERE approvable_type = @approval_type
|
||||
GROUP BY approvable_id
|
||||
) la
|
||||
ON la.approvable_id = a.approvable_id
|
||||
AND la.latest_action_at = a.action_at
|
||||
WHERE a.approvable_type = @approval_type
|
||||
),
|
||||
approved_transfers AS (
|
||||
SELECT
|
||||
lt.id,
|
||||
lt.from_project_flock_id,
|
||||
COALESCE(DATE(lt.effective_move_date), DATE(lt.economic_cutoff_date), DATE(lt.transfer_date)) AS effective_date
|
||||
FROM laying_transfers lt
|
||||
JOIN latest_transfer_approval lta ON lta.approvable_id = lt.id
|
||||
WHERE lt.deleted_at IS NULL
|
||||
AND lt.executed_at IS NOT NULL
|
||||
AND lta.action = 'APPROVED'
|
||||
)
|
||||
SELECT
|
||||
ltt.target_project_flock_kandang_id AS project_flock_kandang_id,
|
||||
at.from_project_flock_id AS source_project_flock_id,
|
||||
at.effective_date AS transfer_date,
|
||||
ltt.total_qty AS transfer_qty,
|
||||
at.id AS transfer_id
|
||||
FROM laying_transfer_targets ltt
|
||||
JOIN approved_transfers at ON at.id = ltt.laying_transfer_id
|
||||
WHERE ltt.deleted_at IS NULL
|
||||
AND ltt.target_project_flock_kandang_id = @project_flock_kandang_id
|
||||
AND at.effective_date <= DATE(@period_date)
|
||||
ORDER BY at.effective_date ASC, at.id ASC
|
||||
`
|
||||
|
||||
err := r.db.WithContext(ctx).Raw(query, map[string]any{
|
||||
"approval_type": utils.ApprovalWorkflowTransferToLaying.String(),
|
||||
"project_flock_kandang_id": projectFlockKandangId,
|
||||
"period_date": period,
|
||||
}).Scan(&rows).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return rows, nil
|
||||
}
|
||||
|
||||
func (r *HppV2RepositoryImpl) GetManualDepreciationInputByProjectFlockID(
|
||||
ctx context.Context,
|
||||
projectFlockID uint,
|
||||
@@ -373,7 +435,34 @@ func (r *HppV2RepositoryImpl) GetEarliestChickInDateByProjectFlockID(ctx context
|
||||
return selected.ChickInDate, nil
|
||||
}
|
||||
|
||||
func (r *HppV2RepositoryImpl) GetDepreciationPercents(
|
||||
func (r *HppV2RepositoryImpl) GetChickinPopulationByPFKForFarm(
|
||||
ctx context.Context,
|
||||
projectFlockID uint,
|
||||
) (map[uint]float64, error) {
|
||||
type row struct {
|
||||
ProjectFlockKandangID uint `gorm:"column:project_flock_kandang_id"`
|
||||
TotalQty float64 `gorm:"column:total_qty"`
|
||||
}
|
||||
var rows []row
|
||||
err := r.db.WithContext(ctx).
|
||||
Table("project_chickins AS pc").
|
||||
Select("pc.project_flock_kandang_id, SUM(pc.usage_qty) AS total_qty").
|
||||
Joins("JOIN project_flock_kandangs AS pfk ON pfk.id = pc.project_flock_kandang_id").
|
||||
Where("pc.deleted_at IS NULL").
|
||||
Where("pfk.project_flock_id = ?", projectFlockID).
|
||||
Group("pc.project_flock_kandang_id").
|
||||
Scan(&rows).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
result := make(map[uint]float64, len(rows))
|
||||
for _, x := range rows {
|
||||
result[x.ProjectFlockKandangID] = x.TotalQty
|
||||
}
|
||||
return result, nil
|
||||
}
|
||||
|
||||
func (r *HppV2RepositoryImpl) GetMultiplicationPercentages(
|
||||
ctx context.Context,
|
||||
houseTypes []string,
|
||||
maxDay int,
|
||||
@@ -384,19 +473,19 @@ func (r *HppV2RepositoryImpl) GetDepreciationPercents(
|
||||
}
|
||||
|
||||
type row struct {
|
||||
HouseType string
|
||||
Day int
|
||||
DepreciationPercent float64
|
||||
HouseType string
|
||||
Day int
|
||||
MultiplicationPercentage float64
|
||||
}
|
||||
|
||||
rows := make([]row, 0)
|
||||
err := r.db.WithContext(ctx).
|
||||
Table("house_depreciation_standards").
|
||||
Select("house_type::text AS house_type, day, depreciation_percent").
|
||||
Where("house_type::text IN ?", houseTypes).
|
||||
Where("day <= ?", maxDay).
|
||||
Order("house_type ASC, day ASC").
|
||||
Scan(&rows).Error
|
||||
err := r.db.WithContext(ctx).Raw(`
|
||||
SELECT DISTINCT ON (house_type::text, day)
|
||||
house_type::text AS house_type, day, multiplication_percentage
|
||||
FROM house_depreciation_standards
|
||||
WHERE house_type::text IN ? AND day <= ?
|
||||
ORDER BY house_type, day, effective_date DESC NULLS LAST
|
||||
`, houseTypes, maxDay).Scan(&rows).Error
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
@@ -405,7 +494,7 @@ func (r *HppV2RepositoryImpl) GetDepreciationPercents(
|
||||
if _, exists := result[item.HouseType]; !exists {
|
||||
result[item.HouseType] = make(map[int]float64)
|
||||
}
|
||||
result[item.HouseType][item.Day] = item.DepreciationPercent
|
||||
result[item.HouseType][item.Day] = item.MultiplicationPercentage
|
||||
}
|
||||
|
||||
return result, nil
|
||||
|
||||
@@ -6,8 +6,8 @@ import (
|
||||
)
|
||||
|
||||
const (
|
||||
depreciationStartAgeDayCloseHouse = 155
|
||||
depreciationStartAgeDayOpenHouse = 176
|
||||
depreciationStartAgeDayCloseHouse = 175
|
||||
depreciationStartAgeDayOpenHouse = 175
|
||||
)
|
||||
|
||||
func NormalizeDepreciationHouseType(raw string) string {
|
||||
@@ -26,12 +26,12 @@ func DepreciationStartAgeDay(houseType string) int {
|
||||
}
|
||||
|
||||
func FlockAgeDay(originDate time.Time, periodDate time.Time) int {
|
||||
origin := time.Date(originDate.Year(), originDate.Month(), originDate.Day(), 0, 0, 0, 0, originDate.Location())
|
||||
period := time.Date(periodDate.Year(), periodDate.Month(), periodDate.Day(), 0, 0, 0, 0, periodDate.Location())
|
||||
origin := time.Date(originDate.Year(), originDate.Month(), originDate.Day(), 0, 0, 0, 0, time.UTC)
|
||||
period := time.Date(periodDate.Year(), periodDate.Month(), periodDate.Day(), 0, 0, 0, 0, time.UTC)
|
||||
if period.Before(origin) {
|
||||
return 0
|
||||
}
|
||||
return int(period.Sub(origin).Hours()/24) + 1
|
||||
return int(period.Sub(origin).Hours() / 24)
|
||||
}
|
||||
|
||||
func DepreciationScheduleDay(originDate time.Time, periodDate time.Time, houseType string) int {
|
||||
@@ -47,9 +47,9 @@ func CalculateDepreciationAtDayN(
|
||||
initialPulletCost float64,
|
||||
dayN int,
|
||||
houseType string,
|
||||
percentByHouseType map[string]map[int]float64,
|
||||
multiplicationByHouseType map[string]map[int]float64,
|
||||
) (float64, float64, float64) {
|
||||
return CalculateDepreciationFromDayRange(initialPulletCost, 1, dayN, houseType, percentByHouseType)
|
||||
return CalculateDepreciationFromDayRange(initialPulletCost, 1, dayN, houseType, multiplicationByHouseType)
|
||||
}
|
||||
|
||||
func CalculateDepreciationFromDayRange(
|
||||
@@ -57,8 +57,8 @@ func CalculateDepreciationFromDayRange(
|
||||
startDay int,
|
||||
endDay int,
|
||||
houseType string,
|
||||
percentByHouseType map[string]map[int]float64,
|
||||
) (float64, float64, float64) {
|
||||
multiplicationByHouseType map[string]map[int]float64,
|
||||
) (pulletCostDayN, depreciationValue, multiplicationPercentage float64) {
|
||||
if initialPulletCost <= 0 || endDay <= 0 {
|
||||
return 0, 0, 0
|
||||
}
|
||||
@@ -70,30 +70,30 @@ func CalculateDepreciationFromDayRange(
|
||||
}
|
||||
|
||||
normalizedHouseType := NormalizeDepreciationHouseType(houseType)
|
||||
housePercent, exists := percentByHouseType[normalizedHouseType]
|
||||
houseMult, exists := multiplicationByHouseType[normalizedHouseType]
|
||||
if !exists {
|
||||
return 0, 0, 0
|
||||
}
|
||||
|
||||
current := initialPulletCost
|
||||
pulletCostDayN := 0.0
|
||||
depreciationValue := 0.0
|
||||
depreciationPercent := 0.0
|
||||
for day := startDay; day <= endDay; day++ {
|
||||
pct := housePercent[day]
|
||||
dep := current * (pct / 100)
|
||||
mult, ok := houseMult[day]
|
||||
if !ok {
|
||||
// No standard for this day → assume no depreciation (mult=1).
|
||||
mult = 1.0
|
||||
}
|
||||
if day == endDay {
|
||||
pulletCostDayN = current
|
||||
depreciationValue = dep
|
||||
depreciationPercent = pct
|
||||
multiplicationPercentage = mult
|
||||
depreciationValue = current * (1.0 - mult)
|
||||
}
|
||||
current -= dep
|
||||
current = current * mult
|
||||
if current < 0 {
|
||||
current = 0
|
||||
}
|
||||
}
|
||||
|
||||
return pulletCostDayN, depreciationValue, depreciationPercent
|
||||
return pulletCostDayN, depreciationValue, multiplicationPercentage
|
||||
}
|
||||
|
||||
func CalculateEffectiveDepreciationPercent(totalDepreciationValue, totalPulletCostDayN float64) float64 {
|
||||
|
||||
@@ -1191,26 +1191,72 @@ func (s *hppV2Service) getDepreciationComponent(
|
||||
}, nil
|
||||
}
|
||||
|
||||
if totalPulletCost <= 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
transferInput, err := s.hppRepo.GetLatestTransferInputByProjectFlockKandangID(context.Background(), projectFlockKandangId, periodDate)
|
||||
// Multi-source support: 1 target kandang bisa menerima dari MULTIPLE transfer terpisah
|
||||
// (tiap transfer = 1 source kandang). Depresiasi per target = SUM dari per-transfer depresiasi.
|
||||
// Setiap transfer dihitung dengan chick_in_date source-nya sendiri dan cost basis pro-rated
|
||||
// berdasarkan qty share (transfer.qty / totalTransferQty).
|
||||
transferInputs, err := s.hppRepo.GetAllTransferInputsByProjectFlockKandangID(context.Background(), projectFlockKandangId, periodDate)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var part *HppV2ComponentPart
|
||||
if transferInput != nil && transferInput.SourceProjectFlockID > 0 {
|
||||
part, err = s.buildNormalTransferDepreciationPart(contextRow, transferInput, periodDate, totalPulletCost)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
// Filter valid transfers (punya source flock id)
|
||||
validTransfers := make([]commonRepo.HppV2LatestTransferInputRow, 0, len(transferInputs))
|
||||
totalTransferQty := 0.0
|
||||
for _, t := range transferInputs {
|
||||
if t.SourceProjectFlockID == 0 {
|
||||
continue
|
||||
}
|
||||
} else {
|
||||
part, err = s.buildManualCutoverDepreciationPart(projectFlockKandangId, contextRow, periodDate, totalPulletCost)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
validTransfers = append(validTransfers, t)
|
||||
totalTransferQty += t.TransferQty
|
||||
}
|
||||
|
||||
if len(validTransfers) > 0 {
|
||||
if totalPulletCost <= 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
totalDepreciation := 0.0
|
||||
parts := make([]HppV2ComponentPart, 0, len(validTransfers))
|
||||
for i := range validTransfers {
|
||||
t := validTransfers[i]
|
||||
// Pro-rate cost basis per transfer berdasarkan qty share.
|
||||
// CATATAN: pendekatan ini AKURAT kalau cost per ekor sama antar source flock.
|
||||
// Kalau cost per ekor berbeda signifikan antar source, follow-up: refactor
|
||||
// `buildGrowingUsagePart` untuk multi-source-flock cost computation.
|
||||
transferCostBasis := totalPulletCost
|
||||
if totalTransferQty > 0 && len(validTransfers) > 1 {
|
||||
transferCostBasis = totalPulletCost * (t.TransferQty / totalTransferQty)
|
||||
}
|
||||
|
||||
part, partErr := s.buildNormalTransferDepreciationPart(contextRow, &t, periodDate, transferCostBasis)
|
||||
if partErr != nil {
|
||||
return nil, partErr
|
||||
}
|
||||
if part == nil {
|
||||
continue
|
||||
}
|
||||
totalDepreciation += part.Total
|
||||
parts = append(parts, *part)
|
||||
}
|
||||
|
||||
if len(parts) == 0 {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
return &HppV2Component{
|
||||
Code: hppV2ComponentDepreciation,
|
||||
Title: "Depreciation",
|
||||
Scopes: []string{hppV2ScopeProductionCost},
|
||||
Total: totalDepreciation,
|
||||
Parts: parts,
|
||||
}, nil
|
||||
}
|
||||
|
||||
// Fallback: manual cut-over (kandang tanpa transfer record)
|
||||
part, err := s.buildManualCutoverDepreciationPart(projectFlockKandangId, contextRow, periodDate, totalPulletCost)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if part == nil {
|
||||
return nil, nil
|
||||
@@ -1344,20 +1390,22 @@ func (s *hppV2Service) buildNormalTransferDepreciationPart(
|
||||
}
|
||||
|
||||
houseType := NormalizeDepreciationHouseType(contextRow.HouseType)
|
||||
percentByHouseType, err := s.hppRepo.GetDepreciationPercents(context.Background(), []string{houseType}, scheduleDay)
|
||||
multiplicationByHouseType, err := s.hppRepo.GetMultiplicationPercentages(context.Background(), []string{houseType}, scheduleDay)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
pulletCostDayN, depreciationValue, depreciationPercent := CalculateDepreciationAtDayN(
|
||||
pulletCostDayN, depreciationValue, multiplicationPercentage := CalculateDepreciationAtDayN(
|
||||
totalPulletCost,
|
||||
scheduleDay,
|
||||
contextRow.HouseType,
|
||||
percentByHouseType,
|
||||
multiplicationByHouseType,
|
||||
)
|
||||
if depreciationValue <= 0 {
|
||||
if depreciationValue <= 0 && pulletCostDayN <= 0 {
|
||||
return nil, nil
|
||||
}
|
||||
totalValueAfter := pulletCostDayN * multiplicationPercentage
|
||||
depreciationPercent := (1.0 - multiplicationPercentage) * 100.0
|
||||
|
||||
return &HppV2ComponentPart{
|
||||
Code: hppV2PartDepreciationNormal,
|
||||
@@ -1365,13 +1413,15 @@ func (s *hppV2Service) buildNormalTransferDepreciationPart(
|
||||
Scopes: []string{hppV2ScopeProductionCost},
|
||||
Total: depreciationValue,
|
||||
Details: map[string]any{
|
||||
"basis_total": totalPulletCost,
|
||||
"pullet_cost_day_n": pulletCostDayN,
|
||||
"depreciation_percent": depreciationPercent,
|
||||
"schedule_day": scheduleDay,
|
||||
"origin_date": formatDateOnly(*originDate),
|
||||
"transfer_date": formatDateOnly(transferInput.TransferDate),
|
||||
"source_project_flock_id": transferInput.SourceProjectFlockID,
|
||||
"basis_total": totalPulletCost,
|
||||
"pullet_cost_day_n": pulletCostDayN,
|
||||
"multiplication_percentage": multiplicationPercentage,
|
||||
"total_value_pullet_after_depreciation": totalValueAfter,
|
||||
"depreciation_percent": depreciationPercent,
|
||||
"schedule_day": scheduleDay,
|
||||
"origin_date": formatDateOnly(*originDate),
|
||||
"transfer_date": formatDateOnly(transferInput.TransferDate),
|
||||
"source_project_flock_id": transferInput.SourceProjectFlockID,
|
||||
},
|
||||
References: []HppV2Reference{
|
||||
{
|
||||
@@ -1392,7 +1442,7 @@ func (s *hppV2Service) buildManualCutoverDepreciationPart(
|
||||
periodDate time.Time,
|
||||
totalPulletCost float64,
|
||||
) (*HppV2ComponentPart, error) {
|
||||
if contextRow == nil || totalPulletCost <= 0 {
|
||||
if contextRow == nil {
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
@@ -1407,6 +1457,21 @@ func (s *hppV2Service) buildManualCutoverDepreciationPart(
|
||||
return nil, nil
|
||||
}
|
||||
|
||||
populations, err := s.hppRepo.GetChickinPopulationByPFKForFarm(context.Background(), contextRow.ProjectFlockID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
var totalPopulation float64
|
||||
for _, qty := range populations {
|
||||
totalPopulation += qty
|
||||
}
|
||||
kandangPopulation := populations[projectFlockKandangId]
|
||||
if totalPopulation <= 0 || kandangPopulation <= 0 {
|
||||
return nil, nil
|
||||
}
|
||||
populationShare := kandangPopulation / totalPopulation
|
||||
basis := manualInput.TotalCost * populationShare
|
||||
|
||||
originDate, err := s.hppRepo.GetEarliestChickInDateByProjectFlockID(context.Background(), contextRow.ProjectFlockID)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
@@ -1427,21 +1492,24 @@ func (s *hppV2Service) buildManualCutoverDepreciationPart(
|
||||
}
|
||||
|
||||
houseType := NormalizeDepreciationHouseType(contextRow.HouseType)
|
||||
percentByHouseType, err := s.hppRepo.GetDepreciationPercents(context.Background(), []string{houseType}, reportScheduleDay)
|
||||
multiplicationByHouseType, err := s.hppRepo.GetMultiplicationPercentages(context.Background(), []string{houseType}, reportScheduleDay)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
pulletCostDayN, depreciationValue, depreciationPercent := CalculateDepreciationFromDayRange(
|
||||
totalPulletCost,
|
||||
pulletCostDayN, depreciationValue, multiplicationPercentage := CalculateDepreciationFromDayRange(
|
||||
basis,
|
||||
startDay,
|
||||
reportScheduleDay,
|
||||
contextRow.HouseType,
|
||||
percentByHouseType,
|
||||
multiplicationByHouseType,
|
||||
)
|
||||
if depreciationValue <= 0 {
|
||||
if depreciationValue <= 0 && pulletCostDayN <= 0 {
|
||||
return nil, nil
|
||||
}
|
||||
totalValueAfter := pulletCostDayN * multiplicationPercentage
|
||||
depreciationPercent := (1.0 - multiplicationPercentage) * 100.0
|
||||
_ = totalPulletCost
|
||||
|
||||
return &HppV2ComponentPart{
|
||||
Code: hppV2PartDepreciationCutover,
|
||||
@@ -1449,15 +1517,19 @@ func (s *hppV2Service) buildManualCutoverDepreciationPart(
|
||||
Scopes: []string{hppV2ScopeProductionCost},
|
||||
Total: depreciationValue,
|
||||
Details: map[string]any{
|
||||
"basis_total": totalPulletCost,
|
||||
"pullet_cost_day_n": pulletCostDayN,
|
||||
"depreciation_percent": depreciationPercent,
|
||||
"schedule_day": reportScheduleDay,
|
||||
"start_schedule_day": startDay,
|
||||
"origin_date": formatDateOnly(*originDate),
|
||||
"cutover_date": formatDateOnly(manualInput.CutoverDate),
|
||||
"manual_input_id": manualInput.ID,
|
||||
"project_flock_kandang": projectFlockKandangId,
|
||||
"basis_total": basis,
|
||||
"manual_input_total": manualInput.TotalCost,
|
||||
"population_share": populationShare,
|
||||
"pullet_cost_day_n": pulletCostDayN,
|
||||
"multiplication_percentage": multiplicationPercentage,
|
||||
"total_value_pullet_after_depreciation": totalValueAfter,
|
||||
"depreciation_percent": depreciationPercent,
|
||||
"schedule_day": reportScheduleDay,
|
||||
"start_schedule_day": startDay,
|
||||
"origin_date": formatDateOnly(*originDate),
|
||||
"cutover_date": formatDateOnly(manualInput.CutoverDate),
|
||||
"manual_input_id": manualInput.ID,
|
||||
"project_flock_kandang": projectFlockKandangId,
|
||||
},
|
||||
References: []HppV2Reference{
|
||||
{
|
||||
@@ -1465,7 +1537,7 @@ func (s *hppV2Service) buildManualCutoverDepreciationPart(
|
||||
ID: manualInput.ID,
|
||||
Date: formatDateOnly(manualInput.CutoverDate),
|
||||
Qty: 1,
|
||||
Total: totalPulletCost,
|
||||
Total: manualInput.TotalCost,
|
||||
AppliedTotal: depreciationValue,
|
||||
},
|
||||
},
|
||||
@@ -1724,7 +1796,7 @@ func partHasScope(part *HppV2ComponentPart, scope string) bool {
|
||||
}
|
||||
|
||||
func dateOnly(value time.Time) time.Time {
|
||||
return time.Date(value.Year(), value.Month(), value.Day(), 0, 0, 0, 0, value.Location())
|
||||
return time.Date(value.Year(), value.Month(), value.Day(), 0, 0, 0, 0, time.UTC)
|
||||
}
|
||||
|
||||
func formatDateOnly(value time.Time) string {
|
||||
|
||||
@@ -57,6 +57,14 @@ func (s *hppV2RepoStub) GetLatestTransferInputByProjectFlockKandangID(_ context.
|
||||
return s.latestTransferByPFK[projectFlockKandangId], nil
|
||||
}
|
||||
|
||||
func (s *hppV2RepoStub) GetAllTransferInputsByProjectFlockKandangID(_ context.Context, projectFlockKandangId uint, _ time.Time) ([]commonRepo.HppV2LatestTransferInputRow, error) {
|
||||
row := s.latestTransferByPFK[projectFlockKandangId]
|
||||
if row == nil {
|
||||
return []commonRepo.HppV2LatestTransferInputRow{}, nil
|
||||
}
|
||||
return []commonRepo.HppV2LatestTransferInputRow{*row}, nil
|
||||
}
|
||||
|
||||
func (s *hppV2RepoStub) GetManualDepreciationInputByProjectFlockID(_ context.Context, projectFlockID uint) (*commonRepo.HppV2ManualDepreciationInputRow, error) {
|
||||
return s.manualInputByProject[projectFlockID], nil
|
||||
}
|
||||
@@ -93,6 +101,18 @@ func (s *hppV2RepoStub) GetDepreciationPercents(_ context.Context, houseTypes []
|
||||
return result, nil
|
||||
}
|
||||
|
||||
// GetMultiplicationPercentages — alias yang sama dengan GetDepreciationPercents untuk match
|
||||
// interface HppV2CostRepository (interface dipakai method name baru ini).
|
||||
func (s *hppV2RepoStub) GetMultiplicationPercentages(ctx context.Context, houseTypes []string, maxDay int) (map[string]map[int]float64, error) {
|
||||
return s.GetDepreciationPercents(ctx, houseTypes, maxDay)
|
||||
}
|
||||
|
||||
// GetChickinPopulationByPFKForFarm — return populasi per PFK dari satu project flock.
|
||||
// Stub minimal: return empty map (depreciation manual cutover tidak di-test di sini).
|
||||
func (s *hppV2RepoStub) GetChickinPopulationByPFKForFarm(_ context.Context, _ uint) (map[uint]float64, error) {
|
||||
return map[uint]float64{}, nil
|
||||
}
|
||||
|
||||
func (s *hppV2RepoStub) ListUsageCostRowsByProductFlags(_ context.Context, projectFlockKandangIDs []uint, flagNames []string, _ *time.Time) ([]commonRepo.HppV2UsageCostRow, error) {
|
||||
return append([]commonRepo.HppV2UsageCostRow{}, s.usageRowsByKey[stubKey(projectFlockKandangIDs, flagNames)]...), nil
|
||||
}
|
||||
|
||||
@@ -121,9 +121,12 @@ func init() {
|
||||
// Redis
|
||||
RedisURL = viper.GetString("REDIS_URL")
|
||||
|
||||
// TransferToLayingGrowingMaxWeek: batas umur (minggu dari chick_in) yang masih boleh ditransfer ke laying.
|
||||
// Disatukan dengan depreciation_start_age_day = 175 hari = 25 minggu, agar konsisten antara batas transfer
|
||||
// dan kapan depresiasi mulai berjalan.
|
||||
TransferToLayingGrowingMaxWeek = viper.GetInt("TRANSFER_TO_LAYING_GROWING_MAX_WEEK")
|
||||
if TransferToLayingGrowingMaxWeek <= 0 {
|
||||
TransferToLayingGrowingMaxWeek = 19
|
||||
TransferToLayingGrowingMaxWeek = 25
|
||||
}
|
||||
|
||||
// Object storage
|
||||
|
||||
+9
@@ -0,0 +1,9 @@
|
||||
BEGIN;
|
||||
|
||||
DROP INDEX IF EXISTS idx_daily_checklists_unique_non_rejected;
|
||||
|
||||
ALTER TABLE daily_checklists
|
||||
ADD CONSTRAINT daily_checklists_date_kandang_category_key
|
||||
UNIQUE (date, kandang_id, category);
|
||||
|
||||
COMMIT;
|
||||
+10
@@ -0,0 +1,10 @@
|
||||
BEGIN;
|
||||
|
||||
ALTER TABLE daily_checklists
|
||||
DROP CONSTRAINT IF EXISTS daily_checklists_date_kandang_category_key;
|
||||
|
||||
CREATE UNIQUE INDEX IF NOT EXISTS idx_daily_checklists_unique_non_rejected
|
||||
ON daily_checklists (date, kandang_id, category)
|
||||
WHERE (status IS NULL OR status <> 'REJECTED');
|
||||
|
||||
COMMIT;
|
||||
+21
@@ -0,0 +1,21 @@
|
||||
UPDATE recordings r
|
||||
SET day = (
|
||||
SELECT (r.record_datetime::date - MIN(pc.chick_in_date)::date)::int + 1
|
||||
FROM project_chickins pc
|
||||
WHERE pc.project_flock_kandang_id = r.project_flock_kandangs_id
|
||||
AND pc.deleted_at IS NULL
|
||||
)
|
||||
WHERE r.deleted_at IS NULL
|
||||
AND (
|
||||
SELECT MIN(pc.chick_in_date)
|
||||
FROM project_chickins pc
|
||||
WHERE pc.project_flock_kandang_id = r.project_flock_kandangs_id
|
||||
AND pc.deleted_at IS NULL
|
||||
) IS NOT NULL;
|
||||
|
||||
ALTER TABLE recordings
|
||||
DROP CONSTRAINT IF EXISTS chk_recordings_day;
|
||||
|
||||
ALTER TABLE recordings
|
||||
ADD CONSTRAINT chk_recordings_day
|
||||
CHECK (day IS NULL OR day >= 1);
|
||||
@@ -0,0 +1,21 @@
|
||||
ALTER TABLE recordings
|
||||
DROP CONSTRAINT IF EXISTS chk_recordings_day;
|
||||
|
||||
ALTER TABLE recordings
|
||||
ADD CONSTRAINT chk_recordings_day
|
||||
CHECK (day IS NULL OR day >= 0);
|
||||
|
||||
UPDATE recordings r
|
||||
SET day = (
|
||||
SELECT (r.record_datetime::date - MIN(pc.chick_in_date)::date)::int
|
||||
FROM project_chickins pc
|
||||
WHERE pc.project_flock_kandang_id = r.project_flock_kandangs_id
|
||||
AND pc.deleted_at IS NULL
|
||||
)
|
||||
WHERE r.deleted_at IS NULL
|
||||
AND (
|
||||
SELECT MIN(pc.chick_in_date)
|
||||
FROM project_chickins pc
|
||||
WHERE pc.project_flock_kandang_id = r.project_flock_kandangs_id
|
||||
AND pc.deleted_at IS NULL
|
||||
) IS NOT NULL;
|
||||
+13
@@ -0,0 +1,13 @@
|
||||
-- Revert chick_in_date Pullet Cikaum 1 (project_flock_kandang_id = 70) -> 23 Maret 2026
|
||||
UPDATE public.project_chickins
|
||||
SET chick_in_date = DATE '2026-03-23',
|
||||
updated_at = NOW()
|
||||
WHERE project_flock_kandang_id = 70
|
||||
AND deleted_at IS NULL;
|
||||
|
||||
-- Revert chick_in_date Pullet Cikaum 2 (project_flock_kandang_id = 71) -> 15 Desember 2025
|
||||
UPDATE public.project_chickins
|
||||
SET chick_in_date = DATE '2025-12-15',
|
||||
updated_at = NOW()
|
||||
WHERE project_flock_kandang_id = 71
|
||||
AND deleted_at IS NULL;
|
||||
@@ -0,0 +1,13 @@
|
||||
-- Update chick_in_date Pullet Cikaum 1 (project_flock_kandang_id = 70) -> 24 Maret 2026
|
||||
UPDATE public.project_chickins
|
||||
SET chick_in_date = DATE '2026-03-24',
|
||||
updated_at = NOW()
|
||||
WHERE project_flock_kandang_id = 70
|
||||
AND deleted_at IS NULL;
|
||||
|
||||
-- Update chick_in_date Pullet Cikaum 2 (project_flock_kandang_id = 71) -> 6 April 2026
|
||||
UPDATE public.project_chickins
|
||||
SET chick_in_date = DATE '2026-04-06',
|
||||
updated_at = NOW()
|
||||
WHERE project_flock_kandang_id = 71
|
||||
AND deleted_at IS NULL;
|
||||
+21
@@ -0,0 +1,21 @@
|
||||
-- Revert: hitung ulang recording.day menggunakan chick_in_date sebelum perubahan
|
||||
-- PFK 70: old chick_in_date = 2026-03-23
|
||||
-- PFK 71: old chick_in_date = 2025-12-15
|
||||
-- Kembalikan constraint chk_recordings_day ke >= 1
|
||||
|
||||
UPDATE recordings r
|
||||
SET day = GREATEST(1, (r.record_datetime::date -
|
||||
CASE r.project_flock_kandangs_id
|
||||
WHEN 70 THEN DATE '2026-03-23'
|
||||
WHEN 71 THEN DATE '2025-12-15'
|
||||
END)::int + 1),
|
||||
updated_at = NOW()
|
||||
WHERE r.project_flock_kandangs_id IN (70, 71)
|
||||
AND r.deleted_at IS NULL;
|
||||
|
||||
ALTER TABLE recordings
|
||||
DROP CONSTRAINT IF EXISTS chk_recordings_day;
|
||||
|
||||
ALTER TABLE recordings
|
||||
ADD CONSTRAINT chk_recordings_day
|
||||
CHECK (day IS NULL OR day >= 1);
|
||||
+23
@@ -0,0 +1,23 @@
|
||||
-- Normalize recording.day untuk Pullet Cikaum 1 & 2
|
||||
-- Setelah migrasi 20260505083754_update_pullet_cikaum_chick_in_date mengubah chick_in_date:
|
||||
-- PFK 70: 2026-03-23 → 2026-03-24 (shift +1 hari)
|
||||
-- PFK 71: 2025-12-15 → 2026-04-06 (shift +112 hari)
|
||||
-- Recording.day perlu dihitung ulang: day = record_datetime::date - chick_in_date::date
|
||||
-- Edge case: PFK 70 punya 1 recording (2026-03-23) sebelum chick_in_date baru → di-clamp ke 0
|
||||
-- Note: constraint chk_recordings_day diubah ke >= 0 karena zero-indexed day
|
||||
|
||||
ALTER TABLE recordings
|
||||
DROP CONSTRAINT IF EXISTS chk_recordings_day;
|
||||
|
||||
ALTER TABLE recordings
|
||||
ADD CONSTRAINT chk_recordings_day
|
||||
CHECK (day IS NULL OR day >= 0);
|
||||
|
||||
UPDATE recordings r
|
||||
SET day = GREATEST(0, (r.record_datetime::date - pc.chick_in_date::date)::int),
|
||||
updated_at = NOW()
|
||||
FROM project_chickins pc
|
||||
WHERE pc.project_flock_kandang_id = r.project_flock_kandangs_id
|
||||
AND pc.deleted_at IS NULL
|
||||
AND r.deleted_at IS NULL
|
||||
AND r.project_flock_kandangs_id IN (70, 71);
|
||||
@@ -0,0 +1,5 @@
|
||||
-- Rollback price adjustment_stock id=531
|
||||
UPDATE adjustment_stocks
|
||||
SET price = 9535,
|
||||
grand_total = ROUND(9000 * 9535, 3)
|
||||
WHERE id = 531 AND adj_number = 'ADJ-00506';
|
||||
@@ -0,0 +1,7 @@
|
||||
-- Fix price adjustment_stock id=531 (ADJ-00506)
|
||||
-- Old: price=9535, grand_total=85,815,000
|
||||
-- New: price=12635, grand_total=113,715,000
|
||||
UPDATE adjustment_stocks
|
||||
SET price = 12635,
|
||||
grand_total = ROUND(9000 * 12635, 3)
|
||||
WHERE id = 531 AND adj_number = 'ADJ-00506';
|
||||
@@ -0,0 +1 @@
|
||||
ALTER TABLE expenses DROP COLUMN is_paid;
|
||||
@@ -0,0 +1 @@
|
||||
ALTER TABLE expenses ADD COLUMN is_paid BOOLEAN NOT NULL DEFAULT FALSE;
|
||||
+5
@@ -0,0 +1,5 @@
|
||||
BEGIN;
|
||||
|
||||
DROP TABLE IF EXISTS daily_checklist_empty_kandangs;
|
||||
|
||||
COMMIT;
|
||||
+60
@@ -0,0 +1,60 @@
|
||||
BEGIN;
|
||||
|
||||
CREATE TABLE daily_checklist_empty_kandangs (
|
||||
id bigint GENERATED ALWAYS AS IDENTITY PRIMARY KEY,
|
||||
daily_checklist_id bigint NOT NULL,
|
||||
kandang_id bigint NOT NULL,
|
||||
start_date date NOT NULL,
|
||||
end_date date NOT NULL,
|
||||
created_by bigint,
|
||||
deleted_by bigint,
|
||||
created_at timestamptz NOT NULL DEFAULT now(),
|
||||
updated_at timestamptz NOT NULL DEFAULT now(),
|
||||
deleted_at timestamptz,
|
||||
|
||||
CONSTRAINT fk_dcek_daily_checklist
|
||||
FOREIGN KEY (daily_checklist_id) REFERENCES daily_checklists(id) ON DELETE CASCADE,
|
||||
CONSTRAINT fk_dcek_kandang
|
||||
FOREIGN KEY (kandang_id) REFERENCES kandangs(id) ON DELETE CASCADE,
|
||||
CONSTRAINT fk_dcek_created_by
|
||||
FOREIGN KEY (created_by) REFERENCES users(id) ON DELETE SET NULL,
|
||||
CONSTRAINT fk_dcek_deleted_by
|
||||
FOREIGN KEY (deleted_by) REFERENCES users(id) ON DELETE SET NULL,
|
||||
CONSTRAINT ck_dcek_range CHECK (end_date >= start_date)
|
||||
);
|
||||
|
||||
CREATE INDEX idx_dcek_kandang_range
|
||||
ON daily_checklist_empty_kandangs (kandang_id, start_date, end_date)
|
||||
WHERE deleted_at IS NULL;
|
||||
|
||||
CREATE UNIQUE INDEX idx_dcek_daily_checklist_unique
|
||||
ON daily_checklist_empty_kandangs (daily_checklist_id)
|
||||
WHERE deleted_at IS NULL;
|
||||
|
||||
INSERT INTO daily_checklist_empty_kandangs (
|
||||
daily_checklist_id, kandang_id, start_date, end_date, created_by, created_at, updated_at
|
||||
)
|
||||
SELECT
|
||||
dc.id,
|
||||
dc.kandang_id,
|
||||
dc.date AS start_date,
|
||||
COALESCE(
|
||||
(SELECT (next_dc.date - INTERVAL '1 day')::date
|
||||
FROM daily_checklists next_dc
|
||||
WHERE next_dc.kandang_id = dc.kandang_id
|
||||
AND next_dc.date > dc.date
|
||||
AND next_dc.category <> 'empty_kandang'
|
||||
AND (next_dc.status IS NULL OR next_dc.status <> 'REJECTED')
|
||||
AND next_dc.deleted_at IS NULL
|
||||
ORDER BY next_dc.date ASC
|
||||
LIMIT 1),
|
||||
dc.date
|
||||
) AS end_date,
|
||||
dc.created_by,
|
||||
dc.created_at,
|
||||
dc.updated_at
|
||||
FROM daily_checklists dc
|
||||
WHERE dc.category = 'empty_kandang'
|
||||
AND dc.deleted_at IS NULL;
|
||||
|
||||
COMMIT;
|
||||
+2
@@ -0,0 +1,2 @@
|
||||
ALTER TABLE customers DROP COLUMN bank_name;
|
||||
ALTER TABLE suppliers DROP COLUMN bank_name;
|
||||
+2
@@ -0,0 +1,2 @@
|
||||
ALTER TABLE customers ADD COLUMN bank_name VARCHAR(100) NOT NULL DEFAULT '';
|
||||
ALTER TABLE suppliers ADD COLUMN bank_name VARCHAR(100);
|
||||
+17
@@ -0,0 +1,17 @@
|
||||
-- Hapus open_house dan close_house rows dengan effective_date baru
|
||||
DELETE FROM house_depreciation_standards
|
||||
WHERE house_type IN ('open_house', 'close_house') AND effective_date = '2026-05-20';
|
||||
|
||||
-- Hapus kolom multiplication_percentage
|
||||
ALTER TABLE house_depreciation_standards DROP COLUMN multiplication_percentage;
|
||||
|
||||
-- Invalidate snapshot cache
|
||||
DELETE FROM farm_depreciation_snapshots;
|
||||
|
||||
-- Kembalikan unique constraint lama
|
||||
ALTER TABLE house_depreciation_standards
|
||||
DROP CONSTRAINT house_depreciation_standards_house_type_day_eff_unique;
|
||||
|
||||
ALTER TABLE house_depreciation_standards
|
||||
ADD CONSTRAINT house_depreciation_standards_house_type_day_unique
|
||||
UNIQUE (house_type, day);
|
||||
+279
@@ -0,0 +1,279 @@
|
||||
-- Drop unique constraint lama (house_type, day) agar bisa support multi effective_date
|
||||
ALTER TABLE house_depreciation_standards
|
||||
DROP CONSTRAINT house_depreciation_standards_house_type_day_unique;
|
||||
|
||||
-- Unique baru: (house_type, day, effective_date)
|
||||
-- NULL dianggap distinct di PostgreSQL → row lama (effective_date NULL) tidak konflik dengan row baru
|
||||
ALTER TABLE house_depreciation_standards
|
||||
ADD CONSTRAINT house_depreciation_standards_house_type_day_eff_unique
|
||||
UNIQUE (house_type, day, effective_date);
|
||||
|
||||
-- Tambah kolom multiplication_percentage (nilai dari baris ke-3 Excel "Depresiasi 25 week.xlsx")
|
||||
ALTER TABLE house_depreciation_standards
|
||||
ADD COLUMN multiplication_percentage numeric(20,15) NOT NULL DEFAULT 0;
|
||||
|
||||
-- Isi multiplication_percentage untuk semua row existing (effective_date IS NULL)
|
||||
-- Value diambil dari row 3 Excel: kolom A=day1 s/d TL=day532
|
||||
UPDATE house_depreciation_standards AS hds
|
||||
SET multiplication_percentage = v.val
|
||||
FROM (VALUES
|
||||
(1,0.997742664),(2,0.997737557),(3,0.997732426),(4,0.997727273),(5,0.997722096),
|
||||
(6,0.997716895),(7,0.99771167),(8,0.997706422),(9,0.997701149),(10,0.997695853),
|
||||
(11,0.997690531),(12,0.9977),(13,0.997679814),(14,0.997674419),(15,0.998),
|
||||
(16,0.997997998),(17,0.997993982),(18,0.99798995),(19,0.997985901),(20,0.997981837),
|
||||
(21,0.997977755),(22,0.997635934),(23,0.997630332),(24,0.997624703),(25,0.997619048),
|
||||
(26,0.997613365),(27,0.997607656),(28,0.997601918),(29,0.997596154),(30,0.997590361),
|
||||
(31,0.997584541),(32,0.997578692),(33,0.997572816),(34,0.99756691),(35,0.997560976),
|
||||
(36,0.997555012),(37,0.99754902),(38,0.997542998),(39,0.997536946),(40,0.997530864),
|
||||
(41,0.997524752),(42,0.99751861),(43,0.997867804),(44,0.997863248),(45,0.997858672),
|
||||
(46,0.997854077),(47,0.997849462),(48,0.997844828),(49,0.997840173),(50,0.997474747),
|
||||
(51,0.997468354),(52,0.997461929),(53,0.997455471),(54,0.99744898),(55,0.997442455),
|
||||
(56,0.997435897),(57,0.997429306),(58,0.99742268),(59,0.997416021),(60,0.997409326),
|
||||
(61,0.997402597),(62,0.997395833),(63,0.997389034),(64,0.997756171),(65,0.997751124),
|
||||
(66,0.997746056),(67,0.997740964),(68,0.997735849),(69,0.997730711),(70,0.99772555),
|
||||
(71,0.997340426),(72,0.997333333),(73,0.997326203),(74,0.997319035),(75,0.997311828),
|
||||
(76,0.997304582),(77,0.9972973),(78,0.99767712),(79,0.99767171),(80,0.99766628),
|
||||
(81,0.99766082),(82,0.99765533),(83,0.99764982),(84,0.997644287),(85,0.997245179),
|
||||
(86,0.997237569),(87,0.997229917),(88,0.997222222),(89,0.997214485),(90,0.997206704),
|
||||
(91,0.99719888),(92,0.997191011),(93,0.997183099),(94,0.997175141),(95,0.997167139),
|
||||
(96,0.997159091),(97,0.997150997),(98,0.997142857),(99,0.997544003),(100,0.997537957),
|
||||
(101,0.99753188),(102,0.997525773),(103,0.997519636),(104,0.997513469),(105,0.99750727),
|
||||
(106,0.997084548),(107,0.997076023),(108,0.997067449),(109,0.997058824),(110,0.997050147),
|
||||
(111,0.99704142),(112,0.997032641),(113,0.99744898),(114,0.997442455),(115,0.997435897),
|
||||
(116,0.997429306),(117,0.99742268),(118,0.997416021),(119,0.997409326),(120,0.996969697),
|
||||
(121,0.996960486),(122,0.99695122),(123,0.996941896),(124,0.996932515),(125,0.996923077),
|
||||
(126,0.99691358),(127,0.997346307),(128,0.997339246),(129,0.997332148),(130,0.997325011),
|
||||
(131,0.997317836),(132,0.997310623),(133,0.997303371),(134,0.996845426),(135,0.996835443),
|
||||
(136,0.996825397),(137,0.996815287),(138,0.996805112),(139,0.996794872),(140,0.996784566),
|
||||
(141,0.997235023),(142,0.997227357),(143,0.997219648),(144,0.997211896),(145,0.997204101),
|
||||
(146,0.997196262),(147,0.997188379),(148,0.996710526),(149,0.99669967),(150,0.996688742),
|
||||
(151,0.996677741),(152,0.996666667),(153,0.996655518),(154,0.996644295),(155,0.997113997),
|
||||
(156,0.997105644),(157,0.997097242),(158,0.997088792),(159,0.997080292),(160,0.997071742),
|
||||
(161,0.997063142),(162,0.997054492),(163,0.99704579),(164,0.997037037),(165,0.997028232),
|
||||
(166,0.997019374),(167,0.997010463),(168,0.997001499),(169,0.996491228),(170,0.996478873),
|
||||
(171,0.996466431),(172,0.996453901),(173,0.996441281),(174,0.996428571),(175,0.996415771),
|
||||
(176,0.996916752),(177,0.996907216),(178,0.996897622),(179,0.996887967),(180,0.996878252),
|
||||
(181,0.996868476),(182,0.996858639),(183,0.996848739),(184,0.996838778),(185,0.996828753),
|
||||
(186,0.996818664),(187,0.996808511),(188,0.996798292),(189,0.996788009),(190,0.996240602),
|
||||
(191,0.996226415),(192,0.996212121),(193,0.996197719),(194,0.996183206),(195,0.996168582),
|
||||
(196,0.996153846),(197,0.996690568),(198,0.996679579),(199,0.996668517),(200,0.996657382),
|
||||
(201,0.996646171),(202,0.996634885),(203,0.996623523),(204,0.996612084),(205,0.996600567),
|
||||
(206,0.996588971),(207,0.996577296),(208,0.996565541),(209,0.996553705),(210,0.996541787),
|
||||
(211,0.996529786),(212,0.996517702),(213,0.996505533),(214,0.996493279),(215,0.996480938),
|
||||
(216,0.996468511),(217,0.996455995),(218,0.996443391),(219,0.996430696),(220,0.99641791),
|
||||
(221,0.996405033),(222,0.996392063),(223,0.996378998),(224,0.996365839),(225,0.995744681),
|
||||
(226,0.995726496),(227,0.995708155),(228,0.995689655),(229,0.995670996),(230,0.995652174),
|
||||
(231,0.995633188),(232,0.996240602),(233,0.996226415),(234,0.996212121),(235,0.996197719),
|
||||
(236,0.996183206),(237,0.996168582),(238,0.996153846),(239,0.9961389960),(240,0.996124031),
|
||||
(241,0.996108949),(242,0.99609375),(243,0.996078431),(244,0.996062992),(245,0.996047431),
|
||||
(246,0.996031746),(247,0.996015936),(248,0.996),(249,0.995983936),(250,0.995967742),
|
||||
(251,0.995951417),(252,0.995934959),(253,0.995918367),(254,0.995901639),(255,0.995884774),
|
||||
(256,0.995867769),(257,0.995850622),(258,0.995833333),(259,0.9958158999),(260,0.995798319),
|
||||
(261,0.995780591),(262,0.995762712),(263,0.995744681),(264,0.995726496),(265,0.995708155),
|
||||
(266,0.995689655),(267,0.995670996),(268,0.995652174),(269,0.995633188),(270,0.995614035),
|
||||
(271,0.995594714),(272,0.995575221),(273,0.995555556),(274,0.995535714),(275,0.995515695),
|
||||
(276,0.995495495),(277,0.995475113),(278,0.995454545),(279,0.99543379),(280,0.995412844),
|
||||
(281,0.995391705),(282,0.99537037),(283,0.995348837),(284,0.995327103),(285,0.995305164),
|
||||
(286,0.995282919),(287,0.995260664),(288,0.996031746),(289,0.996015936),(290,0.996),
|
||||
(291,0.995983936),(292,0.995967742),(293,0.995951417),(294,0.995934959),(295,0.995102041),
|
||||
(296,0.995077933),(297,0.995053586),(298,0.995028998),(299,0.995004163),(300,0.994979079),
|
||||
(301,0.994953743),(302,0.994928149),(303,0.994902294),(304,0.994876174),(305,0.994849785),
|
||||
(306,0.994823123),(307,0.994796184),(308,0.994768963),(309,0.994741455),(310,0.994713656),
|
||||
(311,0.994685562),(312,0.994657168),(313,0.994628469),(314,0.99459946),(315,0.994570136),
|
||||
(316,0.994540491),(317,0.994510522),(318,0.994480221),(319,0.994449584),(320,0.994418605),
|
||||
(321,0.994387278),(322,0.994355597),(323,0.995269631),(324,0.995247148),(325,0.995224451),
|
||||
(326,0.995201536),(327,0.995178399),(328,0.995155039),(329,0.995131451),(330,0.994129159),
|
||||
(331,0.994094488),(332,0.994059406),(333,0.994023904),(334,0.993987976),(335,0.993951613),
|
||||
(336,0.993914807),(337,0.994897959),(338,0.994871795),(339,0.994845361),(340,0.994818653),
|
||||
(341,0.994791667),(342,0.994764398),(343,0.994736842),(344,0.993650794),(345,0.993610224),
|
||||
(346,0.993569132),(347,0.993527508),(348,0.993484342),(349,0.993442623),(350,0.99339934),
|
||||
(351,0.993355482),(352,0.993311037),(353,0.993265993),(354,0.993220339),(355,0.993174061),
|
||||
(356,0.993127148),(357,0.993079585),(358,0.994192799),(359,0.994158879),(360,0.994124559),
|
||||
(361,0.994089835),(362,0.994054697),(363,0.994019139),(364,0.993983153),(365,0.992736077),
|
||||
(366,0.992682927),(367,0.992628993),(368,0.992574257),(369,0.992518703),(370,0.992462312),
|
||||
(371,0.992405063),(372,0.993622449),(373,0.993581515),(374,0.993540052),(375,0.993498049),
|
||||
(376,0.993455497),(377,0.993412385),(378,0.9933687),(379,0.993324433),(380,0.99327957),
|
||||
(381,0.9932341),(382,0.993188011),(383,0.993141289),(384,0.993093923),(385,0.993045897),
|
||||
(386,0.991596639),(387,0.991525424),(388,0.991452991),(389,0.99137931),(390,0.991304348),
|
||||
(391,0.99122807),(392,0.991150442),(393,0.992559524),(394,0.992503748),(395,0.99244713),
|
||||
(396,0.99238965),(397,0.992331288),(398,0.992272025),(399,0.992211838),(400,0.992150706),
|
||||
(401,0.992088608),(402,0.992025518),(403,0.991961415),(404,0.991896272),(405,0.991830065),
|
||||
(406,0.991762768),(407,0.991694352),(408,0.991624791),(409,0.991554054),(410,0.991482112),
|
||||
(411,0.991408935),(412,0.991334489),(413,0.991258741),(414,0.989417989),(415,0.989304813),
|
||||
(416,0.989189189),(417,0.989071038),(418,0.988950276),(419,0.988826816),(420,0.988700565),
|
||||
(421,0.99047619),(422,0.990384615),(423,0.990291262),(424,0.990196078),(425,0.99009901),
|
||||
(426,0.99),(427,0.98989899),(428,0.989795918),(429,0.989690722),(430,0.989583333),
|
||||
(431,0.989473684),(432,0.989361702),(433,0.989247312),(434,0.989130435),(435,0.989010989),
|
||||
(436,0.988888889),(437,0.988764045),(438,0.988636364),(439,0.988505747),(440,0.988372093),
|
||||
(441,0.988235294),(442,0.988095238),(443,0.987951807),(444,0.987804878),(445,0.987654321),
|
||||
(446,0.9875),(447,0.987341772),(448,0.987179487),(449,0.987012987),(450,0.986842105),
|
||||
(451,0.986666667),(452,0.986486486),(453,0.98630137),(454,0.986111111),(455,0.985915493),
|
||||
(456,0.985714286),(457,0.985507246),(458,0.985294118),(459,0.985074627),(460,0.984848485),
|
||||
(461,0.984615385),(462,0.984375),(463,0.987301587),(464,0.987138264),(465,0.986970684),
|
||||
(466,0.98679868),(467,0.986622074),(468,0.986440678),(469,0.986254296),(470,0.982578397),
|
||||
(471,0.982269504),(472,0.981949458),(473,0.981617647),(474,0.981273408),(475,0.980916031),
|
||||
(476,0.980544747),(477,0.98015873),(478,0.979757085),(479,0.979338843),(480,0.978902954),
|
||||
(481,0.978448276),(482,0.977973568),(483,0.977477477),(484,0.976958525),(485,0.976415094),
|
||||
(486,0.975845411),(487,0.975247525),(488,0.974619289),(489,0.973958333),(490,0.973262032),
|
||||
(491,0.978021978),(492,0.97752809),(493,0.977011494),(494,0.976470588),(495,0.975903614),
|
||||
(496,0.975308642),(497,0.974683544),(498,0.967532468),(499,0.966442953),(500,0.965277778),
|
||||
(501,0.964028777),(502,0.962686567),(503,0.96124031),(504,0.959677419),(505,0.966386555),
|
||||
(506,0.965217391),(507,0.963963964),(508,0.962616822),(509,0.961165049),(510,0.95959596),
|
||||
(511,0.957894737),(512,0.945054945),(513,0.941860465),(514,0.938271605),(515,0.934210526),
|
||||
(516,0.929577465),(517,0.924242424),(518,0.918032787),(519,0.928571429),(520,0.923076923),
|
||||
(521,0.916666667),(522,0.909090909),(523,0.9),(524,0.888888889),(525,0.875),
|
||||
(526,0.857142857),(527,0.833333333),(528,0.8),(529,0.75),(530,0.666666667),
|
||||
(531,0.5),(532,9.11e-12)
|
||||
) AS v(day_num, val)
|
||||
WHERE hds.day = v.day_num;
|
||||
|
||||
-- Insert open_house baru dengan effective_date 2026-05-20
|
||||
-- multiplication_percentage diambil dari row existing (sudah di-UPDATE di step sebelumnya)
|
||||
INSERT INTO house_depreciation_standards
|
||||
(house_type, day, effective_date, depreciation_percent, standard_week, name, multiplication_percentage)
|
||||
SELECT
|
||||
'open_house'::house_type_enum,
|
||||
day,
|
||||
'2026-05-20'::date,
|
||||
depreciation_percent,
|
||||
25,
|
||||
'Standard Open House Week 25',
|
||||
multiplication_percentage
|
||||
FROM (
|
||||
SELECT DISTINCT ON (day)
|
||||
day, depreciation_percent, multiplication_percentage
|
||||
FROM house_depreciation_standards
|
||||
WHERE house_type = 'open_house'
|
||||
ORDER BY day, effective_date DESC NULLS LAST
|
||||
) effective_open_house;
|
||||
|
||||
-- Insert close_house baru: depreciation_percent dari open_house, multiplication_percentage dari Excel row 8 (close_house)
|
||||
INSERT INTO house_depreciation_standards
|
||||
(house_type, day, effective_date, depreciation_percent, standard_week, name, multiplication_percentage)
|
||||
SELECT
|
||||
'close_house'::house_type_enum,
|
||||
oh.day,
|
||||
'2026-05-20'::date,
|
||||
oh.depreciation_percent,
|
||||
25,
|
||||
'Standard Close House Week 25',
|
||||
ch.val
|
||||
FROM (
|
||||
SELECT DISTINCT ON (day)
|
||||
day, depreciation_percent
|
||||
FROM house_depreciation_standards
|
||||
WHERE house_type = 'open_house'
|
||||
ORDER BY day, effective_date DESC NULLS LAST
|
||||
) oh
|
||||
JOIN (VALUES
|
||||
(1,0.9981),(2,0.9981),(3,0.9981),(4,0.9981),(5,0.9981),
|
||||
(6,0.9981),(7,0.9981),(8,0.9978),(9,0.9978),(10,0.9978),
|
||||
(11,0.9978),(12,0.9978),(13,0.9978),(14,0.9978),(15,0.9978),
|
||||
(16,0.9978),(17,0.9978),(18,0.9978),(19,0.9978),(20,0.9978),
|
||||
(21,0.9978),(22,0.9981),(23,0.9981),(24,0.9981),(25,0.9981),
|
||||
(26,0.9981),(27,0.9981),(28,0.9981),(29,0.9978),(30,0.9978),
|
||||
(31,0.9978),(32,0.9978),(33,0.9978),(34,0.9978),(35,0.9978),
|
||||
(36,0.9978),(37,0.9978),(38,0.9978),(39,0.9978),(40,0.9978),
|
||||
(41,0.9978),(42,0.9978),(43,0.9978),(44,0.9978),(45,0.9978),
|
||||
(46,0.9978),(47,0.9978),(48,0.9978),(49,0.9978),(50,0.9981),
|
||||
(51,0.9981),(52,0.9981),(53,0.9981),(54,0.9981),(55,0.9981),
|
||||
(56,0.9981),(57,0.9978),(58,0.9978),(59,0.9978),(60,0.9978),
|
||||
(61,0.9978),(62,0.9978),(63,0.9978),(64,0.9978),(65,0.9978),
|
||||
(66,0.9977),(67,0.9977),(68,0.9977),(69,0.9977),(70,0.9977),
|
||||
(71,0.9973),(72,0.9973),(73,0.9973),(74,0.9973),(75,0.9973),
|
||||
(76,0.9973),(77,0.9973),(78,0.9977),(79,0.9977),(80,0.9977),
|
||||
(81,0.9977),(82,0.9977),(83,0.9976),(84,0.9976),(85,0.9972),
|
||||
(86,0.9972),(87,0.9972),(88,0.9972),(89,0.9972),(90,0.9972),
|
||||
(91,0.9972),(92,0.9972),(93,0.9972),(94,0.9972),(95,0.9972),
|
||||
(96,0.9972),(97,0.9972),(98,0.9971),(99,0.9975),(100,0.9975),
|
||||
(101,0.9975),(102,0.9975),(103,0.9975),(104,0.9975),(105,0.9975),
|
||||
(106,0.9971),(107,0.9971),(108,0.9971),(109,0.9971),(110,0.9971),
|
||||
(111,0.997),(112,0.997),(113,0.9974),(114,0.9974),(115,0.9974),
|
||||
(116,0.9974),(117,0.9974),(118,0.9974),(119,0.9974),(120,0.997),
|
||||
(121,0.997),(122,0.997),(123,0.9969),(124,0.9969),(125,0.9969),
|
||||
(126,0.9969),(127,0.9973),(128,0.9973),(129,0.9973),(130,0.9973),
|
||||
(131,0.9973),(132,0.9973),(133,0.9973),(134,0.9968),(135,0.9968),
|
||||
(136,0.9968),(137,0.9968),(138,0.9968),(139,0.9968),(140,0.9968),
|
||||
(141,0.9972),(142,0.9972),(143,0.9972),(144,0.9972),(145,0.9972),
|
||||
(146,0.9972),(147,0.9972),(148,0.9967),(149,0.9967),(150,0.9967),
|
||||
(151,0.9967),(152,0.9967),(153,0.9967),(154,0.9966),(155,0.9971),
|
||||
(156,0.9971),(157,0.9971),(158,0.9971),(159,0.9971),(160,0.9971),
|
||||
(161,0.9971),(162,0.9971),(163,0.997),(164,0.997),(165,0.997),
|
||||
(166,0.997),(167,0.997),(168,0.997),(169,0.9965),(170,0.9965),
|
||||
(171,0.9965),(172,0.9965),(173,0.9964),(174,0.9964),(175,0.9964),
|
||||
(176,0.9969),(177,0.9969),(178,0.9969),(179,0.9969),(180,0.9969),
|
||||
(181,0.9969),(182,0.9969),(183,0.9968),(184,0.9968),(185,0.9968),
|
||||
(186,0.9968),(187,0.9968),(188,0.9968),(189,0.9968),(190,0.9962),
|
||||
(191,0.9962),(192,0.9962),(193,0.9962),(194,0.9962),(195,0.9962),
|
||||
(196,0.9962),(197,0.9967),(198,0.9967),(199,0.9967),(200,0.9967),
|
||||
(201,0.9966),(202,0.9966),(203,0.9966),(204,0.9966),(205,0.9966),
|
||||
(206,0.9966),(207,0.9966),(208,0.9966),(209,0.9966),(210,0.9965),
|
||||
(211,0.9965),(212,0.9965),(213,0.9965),(214,0.9965),(215,0.9965),
|
||||
(216,0.9965),(217,0.9965),(218,0.9964),(219,0.9964),(220,0.9964),
|
||||
(221,0.9964),(222,0.9964),(223,0.9964),(224,0.9964),(225,0.9957),
|
||||
(226,0.9957),(227,0.9957),(228,0.9957),(229,0.9957),(230,0.9957),
|
||||
(231,0.9956),(232,0.9962),(233,0.9962),(234,0.9962),(235,0.9962),
|
||||
(236,0.9962),(237,0.9962),(238,0.9962),(239,0.9961),(240,0.9961),
|
||||
(241,0.9961),(242,0.9961),(243,0.9961),(244,0.9961),(245,0.996),
|
||||
(246,0.996),(247,0.996),(248,0.996),(249,0.996),(250,0.996),
|
||||
(251,0.996),(252,0.9959),(253,0.9959),(254,0.9959),(255,0.9959),
|
||||
(256,0.9959),(257,0.9959),(258,0.9958),(259,0.9958),(260,0.9958),
|
||||
(261,0.9958),(262,0.9958),(263,0.9957),(264,0.9957),(265,0.9957),
|
||||
(266,0.9957),(267,0.9957),(268,0.9957),(269,0.9956),(270,0.9956),
|
||||
(271,0.9956),(272,0.9956),(273,0.9956),(274,0.9955),(275,0.9955),
|
||||
(276,0.9955),(277,0.9955),(278,0.9955),(279,0.9954),(280,0.9954),
|
||||
(281,0.9954),(282,0.9954),(283,0.9953),(284,0.9953),(285,0.9953),
|
||||
(286,0.9953),(287,0.9953),(288,0.996),(289,0.996),(290,0.996),
|
||||
(291,0.996),(292,0.996),(293,0.996),(294,0.9959),(295,0.9951),
|
||||
(296,0.9951),(297,0.9951),(298,0.995),(299,0.995),(300,0.995),
|
||||
(301,0.995),(302,0.9949),(303,0.9949),(304,0.9949),(305,0.9948),
|
||||
(306,0.9948),(307,0.9948),(308,0.9948),(309,0.9947),(310,0.9947),
|
||||
(311,0.9947),(312,0.9947),(313,0.9946),(314,0.9946),(315,0.9946),
|
||||
(316,0.9945),(317,0.9945),(318,0.9945),(319,0.9944),(320,0.9944),
|
||||
(321,0.9944),(322,0.9944),(323,0.9953),(324,0.9952),(325,0.9952),
|
||||
(326,0.9952),(327,0.9952),(328,0.9952),(329,0.9951),(330,0.9941),
|
||||
(331,0.9941),(332,0.9941),(333,0.994),(334,0.994),(335,0.994),
|
||||
(336,0.9939),(337,0.9949),(338,0.9949),(339,0.9948),(340,0.9948),
|
||||
(341,0.9948),(342,0.9948),(343,0.9947),(344,0.9937),(345,0.9936),
|
||||
(346,0.9936),(347,0.9935),(348,0.9935),(349,0.9934),(350,0.9934),
|
||||
(351,0.9934),(352,0.9933),(353,0.9933),(354,0.9932),(355,0.9932),
|
||||
(356,0.9931),(357,0.9931),(358,0.9942),(359,0.9942),(360,0.9941),
|
||||
(361,0.9941),(362,0.9941),(363,0.994),(364,0.994),(365,0.9927),
|
||||
(366,0.9927),(367,0.9926),(368,0.9926),(369,0.9925),(370,0.9925),
|
||||
(371,0.9924),(372,0.9936),(373,0.9936),(374,0.9935),(375,0.9935),
|
||||
(376,0.9935),(377,0.9934),(378,0.9934),(379,0.9933),(380,0.9933),
|
||||
(381,0.9932),(382,0.9932),(383,0.9931),(384,0.9931),(385,0.993),
|
||||
(386,0.9916),(387,0.9915),(388,0.9915),(389,0.9914),(390,0.9913),
|
||||
(391,0.9912),(392,0.9912),(393,0.9926),(394,0.9925),(395,0.9924),
|
||||
(396,0.9924),(397,0.9923),(398,0.9923),(399,0.9922),(400,0.9922),
|
||||
(401,0.9921),(402,0.992),(403,0.992),(404,0.9919),(405,0.9918),
|
||||
(406,0.9918),(407,0.9917),(408,0.9916),(409,0.9916),(410,0.9915),
|
||||
(411,0.9914),(412,0.9913),(413,0.9913),(414,0.9894),(415,0.9893),
|
||||
(416,0.9892),(417,0.9891),(418,0.989),(419,0.9888),(420,0.9887),
|
||||
(421,0.9905),(422,0.9904),(423,0.9903),(424,0.9902),(425,0.9901),
|
||||
(426,0.99),(427,0.9899),(428,0.9898),(429,0.9897),(430,0.9896),
|
||||
(431,0.9895),(432,0.9894),(433,0.9892),(434,0.9891),(435,0.989),
|
||||
(436,0.9889),(437,0.9888),(438,0.9886),(439,0.9885),(440,0.9884),
|
||||
(441,0.9882),(442,0.9881),(443,0.988),(444,0.9878),(445,0.9877),
|
||||
(446,0.9875),(447,0.9873),(448,0.9872),(449,0.987),(450,0.9868),
|
||||
(451,0.9867),(452,0.9865),(453,0.9863),(454,0.9861),(455,0.9859),
|
||||
(456,0.9857),(457,0.9855),(458,0.9853),(459,0.9851),(460,0.9848),
|
||||
(461,0.9846),(462,0.9844),(463,0.9873),(464,0.9871),(465,0.987),
|
||||
(466,0.9868),(467,0.9866),(468,0.9864),(469,0.9863),(470,0.9826),
|
||||
(471,0.9823),(472,0.9819),(473,0.9816),(474,0.9813),(475,0.9809),
|
||||
(476,0.9805),(477,0.9802),(478,0.9798),(479,0.9793),(480,0.9789),
|
||||
(481,0.9784),(482,0.978),(483,0.9775),(484,0.977),(485,0.9764),
|
||||
(486,0.9758),(487,0.9752),(488,0.9746),(489,0.974),(490,0.9733),
|
||||
(491,0.978),(492,0.9775),(493,0.977),(494,0.9765),(495,0.9759),
|
||||
(496,0.9753),(497,0.9747),(498,0.9675),(499,0.9664),(500,0.9653),
|
||||
(501,0.964),(502,0.9627),(503,0.9612),(504,0.9597),(505,0.9664),
|
||||
(506,0.9652),(507,0.964),(508,0.9626),(509,0.9612),(510,0.9596),
|
||||
(511,0.9579),(512,0.9451),(513,0.9419),(514,0.9383),(515,0.9342),
|
||||
(516,0.9296),(517,0.9242),(518,0.918),(519,0.9286),(520,0.9231),
|
||||
(521,0.9167),(522,0.9091),(523,0.9),(524,0.8889),(525,0.875),
|
||||
(526,0.8571),(527,0.8333),(528,0.8),(529,0.75),(530,0.6667),
|
||||
(531,0.5),(532,0)
|
||||
) AS ch(day, val) ON oh.day = ch.day;
|
||||
|
||||
-- Invalidate snapshot cache depreciation agar recompute dengan standard baru
|
||||
DELETE FROM farm_depreciation_snapshots;
|
||||
@@ -0,0 +1,4 @@
|
||||
UPDATE adjustment_stocks
|
||||
SET price = 9535,
|
||||
grand_total = ROUND(8700 * 9535, 3)
|
||||
WHERE id = 532 AND adj_number = 'ADJ-00507';
|
||||
@@ -0,0 +1,5 @@
|
||||
|
||||
UPDATE adjustment_stocks
|
||||
SET price = 12635,
|
||||
grand_total = ROUND(8700 * 12635, 3)
|
||||
WHERE id = 532 AND adj_number = 'ADJ-00507';
|
||||
+31
@@ -0,0 +1,31 @@
|
||||
BEGIN;
|
||||
|
||||
-- Rollback konsolidasi: kembalikan data ke loc 18 / 25 sesuai snapshot pre-migration.
|
||||
-- Order: un-soft-delete locations dulu agar FK tidak gagal saat UPDATE child.
|
||||
|
||||
-- 1. Un-soft-delete locations
|
||||
UPDATE locations SET deleted_at = NULL WHERE id IN (18, 25);
|
||||
|
||||
-- 2. project_flocks: PF 30 -> 18, PF 25 & 31 -> 25
|
||||
UPDATE project_flocks SET location_id = 18, updated_at = NOW() WHERE id = 30;
|
||||
UPDATE project_flocks SET location_id = 25, updated_at = NOW() WHERE id IN (25, 31);
|
||||
|
||||
-- 3. kandangs: K9, K72, K117 -> 18; K10, K73, K116 -> 25
|
||||
UPDATE kandangs SET location_id = 18, updated_at = NOW() WHERE id IN (9, 72, 117);
|
||||
UPDATE kandangs SET location_id = 25, updated_at = NOW() WHERE id IN (10, 73, 116);
|
||||
|
||||
-- 4. kandang_groups: KG 26, 68 -> 18; KG 27, 67 -> 25
|
||||
UPDATE kandang_groups SET location_id = 18, updated_at = NOW() WHERE id IN (26, 68);
|
||||
UPDATE kandang_groups SET location_id = 25, updated_at = NOW() WHERE id IN (27, 67);
|
||||
|
||||
-- 5. warehouses: W27, W145, W152 -> 18; W3, W146, W153 -> 25
|
||||
UPDATE warehouses SET location_id = 18, updated_at = NOW() WHERE id IN (27, 145, 152);
|
||||
UPDATE warehouses SET location_id = 25, updated_at = NOW() WHERE id IN (3, 146, 153);
|
||||
|
||||
-- 6. expenses: list eksplisit per location
|
||||
UPDATE expenses SET location_id = 18, updated_at = NOW()
|
||||
WHERE id IN (36, 345, 500, 501, 502, 503, 504, 505, 506, 507, 508);
|
||||
UPDATE expenses SET location_id = 25, updated_at = NOW()
|
||||
WHERE id IN (9, 37, 509, 510, 511, 512, 513, 514, 515, 516, 517, 518);
|
||||
|
||||
COMMIT;
|
||||
+34
@@ -0,0 +1,34 @@
|
||||
BEGIN;
|
||||
|
||||
-- Konsolidasi 3 lokasi "Pullet Cikaum" jadi 1.
|
||||
-- Pindahkan semua data di loc 18 (Pullet Cikaum 1) & 25 (Pullet Cikaum 2) ke loc 2 (Pullet Cikaum).
|
||||
-- Urutan wajib: semua UPDATE child harus selesai SEBELUM soft-delete locations,
|
||||
-- karena trigger trg_soft_delete_fk_locations akan RAISE EXCEPTION untuk FK
|
||||
-- RESTRICT (project_flocks, kandangs, kandang_groups, expenses) atau SET NULL
|
||||
-- untuk warehouses kalau masih ada child yang reference.
|
||||
|
||||
-- 1. project_flocks (PF 25, 30, 31)
|
||||
UPDATE project_flocks SET location_id = 2, updated_at = NOW()
|
||||
WHERE location_id IN (18, 25);
|
||||
|
||||
-- 2. kandangs (K9, K72, K117, K10, K73, K116)
|
||||
UPDATE kandangs SET location_id = 2, updated_at = NOW()
|
||||
WHERE location_id IN (18, 25);
|
||||
|
||||
-- 3. kandang_groups (KG 26, 68, 27, 67)
|
||||
UPDATE kandang_groups SET location_id = 2, updated_at = NOW()
|
||||
WHERE location_id IN (18, 25);
|
||||
|
||||
-- 4. warehouses (W3, W27, W145, W146, W152, W153)
|
||||
UPDATE warehouses SET location_id = 2, updated_at = NOW()
|
||||
WHERE location_id IN (18, 25);
|
||||
|
||||
-- 5. expenses (23 row BOP)
|
||||
UPDATE expenses SET location_id = 2, updated_at = NOW()
|
||||
WHERE location_id IN (18, 25);
|
||||
|
||||
-- 6. Soft-delete locations 18 & 25 (kosong, aman karena semua child sudah pindah)
|
||||
UPDATE locations SET deleted_at = NOW()
|
||||
WHERE id IN (18, 25) AND deleted_at IS NULL;
|
||||
|
||||
COMMIT;
|
||||
+22
@@ -0,0 +1,22 @@
|
||||
-- Rollback: balik ke rule lama (19 minggu = 133 hari)
|
||||
|
||||
BEGIN;
|
||||
|
||||
UPDATE laying_transfers lt
|
||||
SET economic_cutoff_date = sub.cutoff_date,
|
||||
updated_at = NOW()
|
||||
FROM (
|
||||
SELECT
|
||||
lt2.id AS transfer_id,
|
||||
(MIN(pc.chick_in_date)::date + INTERVAL '133 days')::date AS cutoff_date
|
||||
FROM laying_transfers lt2
|
||||
JOIN project_chickins pc ON pc.project_flock_kandang_id = lt2.source_project_flock_kandang_id
|
||||
WHERE lt2.deleted_at IS NULL
|
||||
AND lt2.source_project_flock_kandang_id IS NOT NULL
|
||||
AND pc.deleted_at IS NULL
|
||||
GROUP BY lt2.id
|
||||
) sub
|
||||
WHERE lt.id = sub.transfer_id
|
||||
AND lt.deleted_at IS NULL;
|
||||
|
||||
COMMIT;
|
||||
+24
@@ -0,0 +1,24 @@
|
||||
-- Recalculate laying_transfers.economic_cutoff_date dari rule 19 minggu (lama) ke 25 minggu (baru,
|
||||
-- sejalan dengan depreciation_start_age_day = 175). Semua transfer historis yang punya
|
||||
-- source_project_flock_kandang_id akan di-update agar economic_cutoff_date = source.chick_in_date + 175 hari.
|
||||
|
||||
BEGIN;
|
||||
|
||||
UPDATE laying_transfers lt
|
||||
SET economic_cutoff_date = sub.cutoff_date,
|
||||
updated_at = NOW()
|
||||
FROM (
|
||||
SELECT
|
||||
lt2.id AS transfer_id,
|
||||
(MIN(pc.chick_in_date)::date + INTERVAL '175 days')::date AS cutoff_date
|
||||
FROM laying_transfers lt2
|
||||
JOIN project_chickins pc ON pc.project_flock_kandang_id = lt2.source_project_flock_kandang_id
|
||||
WHERE lt2.deleted_at IS NULL
|
||||
AND lt2.source_project_flock_kandang_id IS NOT NULL
|
||||
AND pc.deleted_at IS NULL
|
||||
GROUP BY lt2.id
|
||||
) sub
|
||||
WHERE lt.id = sub.transfer_id
|
||||
AND lt.deleted_at IS NULL;
|
||||
|
||||
COMMIT;
|
||||
+3
@@ -0,0 +1,3 @@
|
||||
-- Down migration: tidak ada cara restore TRUNCATE. Snapshot akan auto-regenerate on demand.
|
||||
-- File kosong sengaja: rollback safe karena snapshot dianggap cache yang bisa di-regenerate.
|
||||
SELECT 1;
|
||||
+10
@@ -0,0 +1,10 @@
|
||||
-- Truncate semua farm_depreciation_snapshots agar di-recompute dengan logic baru:
|
||||
-- 1. Multi-transfer per target kandang sekarang menghasilkan multiple parts (1 per transfer)
|
||||
-- 2. Economic cutoff date sudah diupdate dari 19 minggu ke 25 minggu
|
||||
-- 3. Format `components` JSON tetap kompatibel — yang berubah adalah jumlah entries (lebih banyak
|
||||
-- untuk kandang multi-transfer)
|
||||
--
|
||||
-- Snapshot akan otomatis di-regenerate saat user request `/api/reports/expense/depreciation`
|
||||
-- untuk period yang relevan.
|
||||
|
||||
TRUNCATE TABLE farm_depreciation_snapshots;
|
||||
@@ -15,6 +15,7 @@ type Customer struct {
|
||||
Phone string `gorm:"not null;size:20"`
|
||||
Email string `gorm:"type:varchar(50);not null"`
|
||||
AccountNumber string `gorm:"not null;size:50"`
|
||||
BankName string `gorm:"not null;size:100;default:''"`
|
||||
Balance float64 `gorm:"default:0"`
|
||||
CreatedBy uint `gorm:"not null"`
|
||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||
|
||||
@@ -0,0 +1,27 @@
|
||||
package entities
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
type DailyChecklistEmptyKandang struct {
|
||||
Id uint `gorm:"primaryKey"`
|
||||
DailyChecklistId uint `gorm:"not null"`
|
||||
KandangId uint `gorm:"not null"`
|
||||
StartDate time.Time `gorm:"type:date;not null"`
|
||||
EndDate time.Time `gorm:"type:date;not null"`
|
||||
CreatedBy *uint
|
||||
DeletedBy *uint
|
||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||
UpdatedAt time.Time `gorm:"autoUpdateTime"`
|
||||
DeletedAt gorm.DeletedAt `gorm:"index" json:"-"`
|
||||
|
||||
DailyChecklist *DailyChecklist `gorm:"foreignKey:DailyChecklistId;references:Id"`
|
||||
Kandang *KandangGroup `gorm:"foreignKey:KandangId;references:Id"`
|
||||
}
|
||||
|
||||
func (DailyChecklistEmptyKandang) TableName() string {
|
||||
return "daily_checklist_empty_kandangs"
|
||||
}
|
||||
@@ -23,11 +23,12 @@ type DailyChecklist struct {
|
||||
UpdatedAt time.Time `gorm:"autoUpdateTime"`
|
||||
DeletedAt gorm.DeletedAt `gorm:"index" json:"-"`
|
||||
|
||||
Kandang KandangGroup `gorm:"foreignKey:KandangId;references:Id"`
|
||||
Checklist *Checklist `gorm:"foreignKey:ChecklistId;references:Id"`
|
||||
Creator *User `gorm:"foreignKey:CreatedBy;references:Id"`
|
||||
Deleter *User `gorm:"foreignKey:DeletedBy;references:Id"`
|
||||
Tasks []DailyChecklistTask `gorm:"foreignKey:DailyChecklistId;references:Id"`
|
||||
Kandang KandangGroup `gorm:"foreignKey:KandangId;references:Id"`
|
||||
Checklist *Checklist `gorm:"foreignKey:ChecklistId;references:Id"`
|
||||
Creator *User `gorm:"foreignKey:CreatedBy;references:Id"`
|
||||
Deleter *User `gorm:"foreignKey:DeletedBy;references:Id"`
|
||||
Tasks []DailyChecklistTask `gorm:"foreignKey:DailyChecklistId;references:Id"`
|
||||
EmptyKandang *DailyChecklistEmptyKandang `gorm:"foreignKey:DailyChecklistId;references:Id"`
|
||||
}
|
||||
|
||||
type DailyChecklistPhase struct {
|
||||
|
||||
@@ -17,6 +17,7 @@ type Expense struct {
|
||||
RealizationDate time.Time `gorm:"type:date;column:realization_date"`
|
||||
TransactionDate time.Time `gorm:"type:date;not null"`
|
||||
Notes string `gorm:"type:text;column:notes"`
|
||||
IsPaid bool `gorm:"column:is_paid;not null;default:false"`
|
||||
CreatedBy uint64 `gorm:""`
|
||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||
UpdatedAt time.Time `gorm:"autoUpdateTime"`
|
||||
|
||||
@@ -43,6 +43,7 @@ type Recording struct {
|
||||
StandardEggMass *float64 `gorm:"-"`
|
||||
StandardEggWeight *float64 `gorm:"-"`
|
||||
StandardFcr *float64 `gorm:"-"`
|
||||
StandardWeek *int `gorm:"-"`
|
||||
PopulationCanChange *bool `gorm:"-"`
|
||||
TransferExecuted *bool `gorm:"-"`
|
||||
IsTransition *bool `gorm:"-"`
|
||||
|
||||
@@ -19,6 +19,7 @@ type Supplier struct {
|
||||
Address string `gorm:"not null"`
|
||||
Npwp *string `gorm:"size:50"`
|
||||
AccountNumber *string `gorm:"size:50"`
|
||||
BankName *string `gorm:"size:100"`
|
||||
Balance float64 `gorm:"type:numeric(15,3);default:0"`
|
||||
DueDate int `gorm:"not null"`
|
||||
CreatedBy uint `gorm:"not null"`
|
||||
|
||||
@@ -66,6 +66,7 @@ const (
|
||||
P_ProductStockGetOne = "lti.inventory.product_stock.detail"
|
||||
P_ProductWarehousekGetAll = "lti.inventory.product_warehouses.list"
|
||||
P_ProductWarehouseGetOne = "lti.inventory.product_warehouses.detail"
|
||||
P_StockLogGetAll = "lti.inventory.stock_log.list"
|
||||
)
|
||||
const (
|
||||
P_ClosingGetAll = "lti.closing.list"
|
||||
@@ -207,6 +208,7 @@ const (
|
||||
P_Finances_Transaction_DeleteOne = "lti.finance.transactions.delete"
|
||||
)
|
||||
const (
|
||||
P_ChickinsGetAll = "lti.production.chickins.list"
|
||||
P_ChickinsCreateOne = "lti.production.chickins.create"
|
||||
P_ChickinsGetOne = "lti.production.chickins.detail"
|
||||
P_ChickinsApproval = "lti.production.chickins.approve"
|
||||
|
||||
@@ -238,17 +238,17 @@ func (u *DailyChecklistController) GetReport(c *fiber.Ctx) error {
|
||||
}
|
||||
|
||||
result, totalResults, err := u.DailyChecklistService.GetReport(c, query)
|
||||
withoutActivities := func(src map[string]int) map[string]int {
|
||||
if src == nil {
|
||||
return map[string]int{}
|
||||
}
|
||||
return src
|
||||
}
|
||||
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
withoutActivities := func(src map[string]any) map[string]any {
|
||||
if src == nil {
|
||||
return map[string]any{}
|
||||
}
|
||||
return src
|
||||
}
|
||||
|
||||
responseData := make([]dto.DailyChecklistReportDTO, len(result))
|
||||
for i, item := range result {
|
||||
responseData[i] = dto.DailyChecklistReportDTO{
|
||||
@@ -412,6 +412,33 @@ func (u *DailyChecklistController) UpdateOne(c *fiber.Ctx) error {
|
||||
})
|
||||
}
|
||||
|
||||
func (u *DailyChecklistController) UpdateByPut(c *fiber.Ctx) error {
|
||||
req := new(validation.Create)
|
||||
param := c.Params("idDailyChecklist")
|
||||
|
||||
id, err := strconv.Atoi(param)
|
||||
if err != nil {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid Id")
|
||||
}
|
||||
|
||||
if err := c.BodyParser(req); err != nil {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid request body")
|
||||
}
|
||||
|
||||
result, err := u.DailyChecklistService.UpdateByPut(c, req, uint(id))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return c.Status(fiber.StatusOK).
|
||||
JSON(response.Success{
|
||||
Code: fiber.StatusOK,
|
||||
Status: "success",
|
||||
Message: "Update dailyChecklist successfully",
|
||||
Data: dto.ToDailyChecklistListDTO(*result),
|
||||
})
|
||||
}
|
||||
|
||||
func validateDailyChecklistDocumentSizes(files []*multipart.FileHeader) error {
|
||||
const maxDailyChecklistDocumentBytes = 5 * 1024 * 1024 // 5MB
|
||||
for _, file := range files {
|
||||
|
||||
@@ -42,6 +42,13 @@ type DailyChecklistDetailDTO struct {
|
||||
TotalActivity int `json:"total_activity"`
|
||||
Progress float64 `json:"progress"`
|
||||
DocumentURLs []DailyChecklistDocumentDTO `json:"document_urls"`
|
||||
EmptyKandang *DailyChecklistEmptyKandangDTO `json:"empty_kandang,omitempty"`
|
||||
}
|
||||
|
||||
type DailyChecklistEmptyKandangDTO struct {
|
||||
Id uint `json:"id"`
|
||||
StartDate time.Time `json:"start_date"`
|
||||
EndDate time.Time `json:"end_date"`
|
||||
}
|
||||
|
||||
type DailyChecklistDocumentDTO struct {
|
||||
@@ -72,13 +79,14 @@ type DailyChecklistPerformanceOverviewDTO struct {
|
||||
ActivityLeft int `json:"activity_left"`
|
||||
}
|
||||
|
||||
|
||||
type DailyChecklistReportDTO struct {
|
||||
Area DailyChecklistReportEntityDTO `json:"area"`
|
||||
Farm DailyChecklistReportEntityDTO `json:"farm"`
|
||||
Kandang DailyChecklistReportEntityDTO `json:"kandang"`
|
||||
ABK DailyChecklistReportEntityDTO `json:"abk"`
|
||||
Phase string `json:"phase"`
|
||||
DailyActivities map[string]int `json:"daily_activities"`
|
||||
DailyActivities map[string]any `json:"daily_activities"`
|
||||
Summary DailyChecklistReportSummaryDTO `json:"summary"`
|
||||
}
|
||||
|
||||
@@ -179,6 +187,17 @@ func ToDailyChecklistListDTO(e entity.DailyChecklist) DailyChecklistListDTO {
|
||||
}
|
||||
}
|
||||
|
||||
func ToDailyChecklistEmptyKandangDTO(e *entity.DailyChecklistEmptyKandang) *DailyChecklistEmptyKandangDTO {
|
||||
if e == nil || e.Id == 0 {
|
||||
return nil
|
||||
}
|
||||
return &DailyChecklistEmptyKandangDTO{
|
||||
Id: e.Id,
|
||||
StartDate: e.StartDate,
|
||||
EndDate: e.EndDate,
|
||||
}
|
||||
}
|
||||
|
||||
func ToDailyChecklistDetailDTO(checklist entity.DailyChecklist, phases []entity.DailyChecklistPhase, tasks []entity.DailyChecklistActivityTask, assignedEmployees []entity.Employee, totalActivities int, progress float64, documentURLs []DailyChecklistDocumentDTO) DailyChecklistDetailDTO {
|
||||
phaseDTOs := make([]DailyChecklistPhaseDTO, 0, len(phases))
|
||||
for _, phase := range phases {
|
||||
@@ -240,5 +259,6 @@ func ToDailyChecklistDetailDTO(checklist entity.DailyChecklist, phases []entity.
|
||||
TotalActivity: totalActivities,
|
||||
Progress: progress,
|
||||
DocumentURLs: documentURLs,
|
||||
EmptyKandang: ToDailyChecklistEmptyKandangDTO(checklist.EmptyKandang),
|
||||
}
|
||||
}
|
||||
|
||||
@@ -22,6 +22,7 @@ type DailyChecklistModule struct{}
|
||||
|
||||
func (DailyChecklistModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
||||
dailyChecklistRepo := rDailyChecklist.NewDailyChecklistRepository(db)
|
||||
emptyKandangRepo := rDailyChecklist.NewDailyChecklistEmptyKandangRepository(db)
|
||||
phasesRepo := rPhases.NewPhasesRepository(db)
|
||||
userRepo := rUser.NewUserRepository(db)
|
||||
documentRepo := commonRepo.NewDocumentRepository(db)
|
||||
@@ -30,7 +31,7 @@ func (DailyChecklistModule) RegisterRoutes(router fiber.Router, db *gorm.DB, val
|
||||
panic(fmt.Sprintf("failed to create document service: %v", err))
|
||||
}
|
||||
|
||||
dailyChecklistService := sDailyChecklist.NewDailyChecklistService(dailyChecklistRepo, phasesRepo, validate, documentSvc)
|
||||
dailyChecklistService := sDailyChecklist.NewDailyChecklistService(dailyChecklistRepo, emptyKandangRepo, phasesRepo, validate, documentSvc)
|
||||
userService := sUser.NewUserService(userRepo, validate)
|
||||
|
||||
DailyChecklistRoutes(router, userService, dailyChecklistService)
|
||||
|
||||
+98
@@ -0,0 +1,98 @@
|
||||
package repository
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"time"
|
||||
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
type DailyChecklistEmptyKandangRepository interface {
|
||||
repository.BaseRepository[entity.DailyChecklistEmptyKandang]
|
||||
FindByDailyChecklistID(ctx context.Context, dailyChecklistID uint) (*entity.DailyChecklistEmptyKandang, error)
|
||||
FindOverlapping(ctx context.Context, kandangID uint, startDate, endDate time.Time, excludeDailyChecklistID uint) (*entity.DailyChecklistEmptyKandang, error)
|
||||
FindActiveCoveringDate(ctx context.Context, kandangID uint, date time.Time) (*entity.DailyChecklistEmptyKandang, error)
|
||||
FindOverlappingInRange(ctx context.Context, kandangIDs []uint, rangeStart, rangeEnd time.Time) ([]entity.DailyChecklistEmptyKandang, error)
|
||||
SoftDeleteByDailyChecklistID(ctx context.Context, dailyChecklistID uint, actorID *uint) error
|
||||
}
|
||||
|
||||
type DailyChecklistEmptyKandangRepositoryImpl struct {
|
||||
*repository.BaseRepositoryImpl[entity.DailyChecklistEmptyKandang]
|
||||
}
|
||||
|
||||
func NewDailyChecklistEmptyKandangRepository(db *gorm.DB) DailyChecklistEmptyKandangRepository {
|
||||
return &DailyChecklistEmptyKandangRepositoryImpl{
|
||||
BaseRepositoryImpl: repository.NewBaseRepository[entity.DailyChecklistEmptyKandang](db),
|
||||
}
|
||||
}
|
||||
|
||||
func (r *DailyChecklistEmptyKandangRepositoryImpl) FindByDailyChecklistID(ctx context.Context, dailyChecklistID uint) (*entity.DailyChecklistEmptyKandang, error) {
|
||||
var rec entity.DailyChecklistEmptyKandang
|
||||
if err := r.DB().WithContext(ctx).
|
||||
Where("daily_checklist_id = ?", dailyChecklistID).
|
||||
First(&rec).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil, nil
|
||||
}
|
||||
return nil, err
|
||||
}
|
||||
return &rec, nil
|
||||
}
|
||||
|
||||
func (r *DailyChecklistEmptyKandangRepositoryImpl) FindOverlapping(ctx context.Context, kandangID uint, startDate, endDate time.Time, excludeDailyChecklistID uint) (*entity.DailyChecklistEmptyKandang, error) {
|
||||
var rec entity.DailyChecklistEmptyKandang
|
||||
query := r.DB().WithContext(ctx).
|
||||
Where("kandang_id = ? AND start_date <= ? AND end_date >= ?", kandangID, endDate, startDate)
|
||||
if excludeDailyChecklistID > 0 {
|
||||
query = query.Where("daily_checklist_id <> ?", excludeDailyChecklistID)
|
||||
}
|
||||
if err := query.First(&rec).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil, nil
|
||||
}
|
||||
return nil, err
|
||||
}
|
||||
return &rec, nil
|
||||
}
|
||||
|
||||
func (r *DailyChecklistEmptyKandangRepositoryImpl) FindActiveCoveringDate(ctx context.Context, kandangID uint, date time.Time) (*entity.DailyChecklistEmptyKandang, error) {
|
||||
var rec entity.DailyChecklistEmptyKandang
|
||||
if err := r.DB().WithContext(ctx).
|
||||
Where("kandang_id = ? AND start_date <= ? AND end_date >= ?", kandangID, date, date).
|
||||
First(&rec).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil, nil
|
||||
}
|
||||
return nil, err
|
||||
}
|
||||
return &rec, nil
|
||||
}
|
||||
|
||||
func (r *DailyChecklistEmptyKandangRepositoryImpl) FindOverlappingInRange(ctx context.Context, kandangIDs []uint, rangeStart, rangeEnd time.Time) ([]entity.DailyChecklistEmptyKandang, error) {
|
||||
if len(kandangIDs) == 0 {
|
||||
return []entity.DailyChecklistEmptyKandang{}, nil
|
||||
}
|
||||
var recs []entity.DailyChecklistEmptyKandang
|
||||
if err := r.DB().WithContext(ctx).
|
||||
Where("kandang_id IN ? AND start_date <= ? AND end_date >= ?", kandangIDs, rangeEnd, rangeStart).
|
||||
Find(&recs).Error; err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return recs, nil
|
||||
}
|
||||
|
||||
func (r *DailyChecklistEmptyKandangRepositoryImpl) SoftDeleteByDailyChecklistID(ctx context.Context, dailyChecklistID uint, actorID *uint) error {
|
||||
updates := map[string]any{
|
||||
"deleted_at": time.Now(),
|
||||
}
|
||||
if actorID != nil {
|
||||
updates["deleted_by"] = *actorID
|
||||
}
|
||||
return r.DB().WithContext(ctx).
|
||||
Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("daily_checklist_id = ? AND deleted_at IS NULL", dailyChecklistID).
|
||||
Updates(updates).Error
|
||||
}
|
||||
@@ -59,6 +59,7 @@ func DailyChecklistRoutes(v1 fiber.Router, u user.UserService, s dailyChecklist.
|
||||
route.Post("/assignment", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.UpdateAssignment)
|
||||
|
||||
route.Patch("/bulk-update", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.BulkUpdate)
|
||||
route.Put("/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.UpdateByPut)
|
||||
route.Patch("/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.UpdateOne)
|
||||
route.Delete("/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.DeleteOne)
|
||||
}
|
||||
|
||||
@@ -29,6 +29,7 @@ type DailyChecklistService interface {
|
||||
GetOne(ctx *fiber.Ctx, id uint) (*entity.DailyChecklist, error)
|
||||
CreateOne(ctx *fiber.Ctx, req *validation.Create) (*entity.DailyChecklist, error)
|
||||
UpdateOne(ctx *fiber.Ctx, req *validation.Update, id uint) (*entity.DailyChecklist, error)
|
||||
UpdateByPut(ctx *fiber.Ctx, req *validation.Create, id uint) (*entity.DailyChecklist, error)
|
||||
BulkUpdate(ctx *fiber.Ctx, req *validation.BulkStatusUpdate) ([]entity.DailyChecklist, error)
|
||||
DeleteOne(ctx *fiber.Ctx, id uint) error
|
||||
AssignPhases(ctx *fiber.Ctx, id uint, req *validation.AssignPhases) error
|
||||
@@ -43,11 +44,12 @@ type DailyChecklistService interface {
|
||||
}
|
||||
|
||||
type dailyChecklistService struct {
|
||||
Log *logrus.Logger
|
||||
Validate *validator.Validate
|
||||
Repository repository.DailyChecklistRepository
|
||||
PhaseRepo phaseRepo.PhasesRepository
|
||||
DocumentSvc commonSvc.DocumentService
|
||||
Log *logrus.Logger
|
||||
Validate *validator.Validate
|
||||
Repository repository.DailyChecklistRepository
|
||||
EmptyKandangRepo repository.DailyChecklistEmptyKandangRepository
|
||||
PhaseRepo phaseRepo.PhasesRepository
|
||||
DocumentSvc commonSvc.DocumentService
|
||||
}
|
||||
|
||||
type DailyChecklistDocument struct {
|
||||
@@ -104,7 +106,7 @@ type DailyChecklistReportItem struct {
|
||||
EmployeeID uint
|
||||
EmployeeName string
|
||||
PhaseName string
|
||||
DailyActivities map[string]int
|
||||
DailyActivities map[string]any
|
||||
Summary DailyChecklistReportSummary
|
||||
}
|
||||
|
||||
@@ -123,26 +125,30 @@ type DailyChecklistReportCategory struct {
|
||||
}
|
||||
|
||||
const (
|
||||
dailyChecklistDateLayout = "2006-01-02"
|
||||
dailyChecklistCategoryEmptyKandang = "empty_kandang"
|
||||
dailyChecklistStatusRejected = "REJECTED"
|
||||
dailyChecklistStatusDraft = "DRAFT"
|
||||
dailyChecklistErrEmptyKandangExist = "DailyChecklist cannot be created because empty_kandang already exists for at least one date in range"
|
||||
dailyChecklistErrDateOverlapExist = "DailyChecklist cannot be created because at least one date in range already has a checklist"
|
||||
dailyChecklistDateLayout = "2006-01-02"
|
||||
dailyChecklistCategoryEmptyKandang = "empty_kandang"
|
||||
dailyChecklistStatusRejected = "REJECTED"
|
||||
dailyChecklistStatusDraft = "DRAFT"
|
||||
dailyChecklistErrDateOverlapExist = "DailyChecklist cannot be created because at least one date in range already has a checklist"
|
||||
dailyChecklistErrDeletedNonEmptyKandangExists = "DailyChecklist cannot be created as empty_kandang because a deleted non-empty_kandang checklist exists for this date"
|
||||
dailyChecklistErrEmptyKandangRangeOverlap = "Empty kandang range overlaps with an existing empty kandang period for this kandang"
|
||||
dailyChecklistErrDateInsideEmptyKandang = "Tanggal berada dalam periode kandang kosong untuk kandang ini"
|
||||
dailyChecklistErrEmptyKandangEndDateInvalid = "empty_kandang_end_date harus >= date"
|
||||
)
|
||||
|
||||
func NewDailyChecklistService(repo repository.DailyChecklistRepository, phaseRepo phaseRepo.PhasesRepository, validate *validator.Validate, documentSvc commonSvc.DocumentService) DailyChecklistService {
|
||||
func NewDailyChecklistService(repo repository.DailyChecklistRepository, emptyKandangRepo repository.DailyChecklistEmptyKandangRepository, phaseRepo phaseRepo.PhasesRepository, validate *validator.Validate, documentSvc commonSvc.DocumentService) DailyChecklistService {
|
||||
return &dailyChecklistService{
|
||||
Log: utils.Log,
|
||||
Validate: validate,
|
||||
Repository: repo,
|
||||
PhaseRepo: phaseRepo,
|
||||
DocumentSvc: documentSvc,
|
||||
Log: utils.Log,
|
||||
Validate: validate,
|
||||
Repository: repo,
|
||||
EmptyKandangRepo: emptyKandangRepo,
|
||||
PhaseRepo: phaseRepo,
|
||||
DocumentSvc: documentSvc,
|
||||
}
|
||||
}
|
||||
|
||||
func (s dailyChecklistService) withRelations(db *gorm.DB) *gorm.DB {
|
||||
return db.Preload("Kandang")
|
||||
return db.Preload("Kandang").Preload("EmptyKandang")
|
||||
}
|
||||
|
||||
func (s dailyChecklistService) ensureChecklistAccess(c *fiber.Ctx, checklistID uint) error {
|
||||
@@ -276,7 +282,11 @@ func (s dailyChecklistService) GetAll(c *fiber.Ctx, params *validation.Query) ([
|
||||
normalizedSearch := re.ReplaceAllString(params.Search, "")
|
||||
if normalizedSearch != "" {
|
||||
like := "%" + normalizedSearch + "%"
|
||||
db = db.Where("(regexp_replace(k.name, '[^a-zA-Z0-9]', '', 'g') ILIKE ? OR regexp_replace(dc.category::text, '[^a-zA-Z0-9]', '', 'g') ILIKE ?)", like, like)
|
||||
db = db.Where(`(
|
||||
regexp_replace(k.name, '[^a-zA-Z0-9]', '', 'g') ILIKE ? OR
|
||||
regexp_replace(dc.category::text, '[^a-zA-Z0-9]', '', 'g') ILIKE ? OR
|
||||
(dc.category = 'empty_kandang' AND regexp_replace('Kandang Kosong', '[^a-zA-Z0-9]', '', 'g') ILIKE ?)
|
||||
)`, like, like, like)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -519,24 +529,28 @@ func (s *dailyChecklistService) CreateOne(c *fiber.Ctx, req *validation.Create)
|
||||
|
||||
status := req.Status
|
||||
category := req.Category
|
||||
endDate := date
|
||||
|
||||
if req.EmptyKandang {
|
||||
if strings.TrimSpace(req.EmptyKandangEndDate) == "" {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "empty_kandang_end_date is required when empty_kandang is true")
|
||||
}
|
||||
|
||||
endDate, err = time.Parse(dailyChecklistDateLayout, strings.TrimSpace(req.EmptyKandangEndDate))
|
||||
if err != nil {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "invalid empty_kandang_end_date format, use YYYY-MM-DD")
|
||||
}
|
||||
if endDate.Before(date) {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "empty_kandang_end_date must be greater than or equal to date")
|
||||
}
|
||||
|
||||
category = dailyChecklistCategoryEmptyKandang
|
||||
}
|
||||
|
||||
var emptyEndDate time.Time
|
||||
if category == dailyChecklistCategoryEmptyKandang {
|
||||
trimmedEnd := strings.TrimSpace(req.EmptyKandangEndDate)
|
||||
if trimmedEnd == "" {
|
||||
emptyEndDate = date
|
||||
} else {
|
||||
parsedEnd, parseErr := time.Parse(dailyChecklistDateLayout, trimmedEnd)
|
||||
if parseErr != nil {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "invalid empty_kandang_end_date format, use YYYY-MM-DD")
|
||||
}
|
||||
if parsedEnd.Before(date) {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, dailyChecklistErrEmptyKandangEndDateInvalid)
|
||||
}
|
||||
emptyEndDate = parsedEnd
|
||||
}
|
||||
}
|
||||
|
||||
targetID := uint(0)
|
||||
|
||||
err = s.Repository.DB().WithContext(c.Context()).Transaction(func(tx *gorm.DB) error {
|
||||
@@ -544,18 +558,40 @@ func (s *dailyChecklistService) CreateOne(c *fiber.Ctx, req *validation.Create)
|
||||
return err
|
||||
}
|
||||
|
||||
if req.EmptyKandang {
|
||||
if err := s.validateNoChecklistOverlapForEmptyKandang(tx, req.KandangId, date, endDate); err != nil {
|
||||
if category == dailyChecklistCategoryEmptyKandang {
|
||||
if err := s.validateNoNormalChecklistInRange(tx, req.KandangId, date, emptyEndDate, 0); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := s.validateNoEmptyKandangRangeOverlap(tx, req.KandangId, date, emptyEndDate, 0); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := s.validateNoExistingEmptyKandangInRange(tx, req.KandangId, date, emptyEndDate, 0); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := s.validateNoDeletedNonEmptyKandangForDate(tx, req.KandangId, date); err != nil {
|
||||
return err
|
||||
}
|
||||
} else {
|
||||
if err := s.validateDateNotInEmptyKandangRange(tx, req.KandangId, date, 0); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := s.validateDateNotInExistingEmptyKandangChecklist(tx, req.KandangId, date, 0); err != nil {
|
||||
return err
|
||||
}
|
||||
return s.createBulkDailyChecklists(tx, req.KandangId, date, endDate, category, status, &targetID)
|
||||
}
|
||||
|
||||
if err := s.validateNoEmptyKandangConflict(tx, req.KandangId, date, endDate); err != nil {
|
||||
if err := s.createOrReuseSingleDailyChecklist(tx, req.KandangId, date, category, status, &targetID); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return s.createOrReuseSingleDailyChecklist(tx, req.KandangId, date, category, status, &targetID)
|
||||
if category == dailyChecklistCategoryEmptyKandang {
|
||||
actorID, _ := m.ActorIDFromContext(c)
|
||||
if err := s.upsertEmptyKandangRange(tx, targetID, req.KandangId, date, emptyEndDate, actorID); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
s.Log.Errorf("Failed to create/upsert dailyChecklist: %+v", err)
|
||||
@@ -585,31 +621,131 @@ func (s *dailyChecklistService) lockKandangForChecklistCreation(tx *gorm.DB, kan
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateNoChecklistOverlapForEmptyKandang(tx *gorm.DB, kandangID uint, startDate, endDate time.Time) error {
|
||||
func (s *dailyChecklistService) validateNoNormalChecklistInRange(tx *gorm.DB, kandangID uint, startDate, endDate time.Time, excludeDCID uint) error {
|
||||
q := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("kandang_id = ? AND date BETWEEN ? AND ? AND category <> ? AND deleted_at IS NULL",
|
||||
kandangID, startDate, endDate, dailyChecklistCategoryEmptyKandang)
|
||||
if excludeDCID > 0 {
|
||||
q = q.Where("id <> ?", excludeDCID)
|
||||
}
|
||||
var conflictCount int64
|
||||
if err := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("kandang_id = ? AND date BETWEEN ? AND ? AND deleted_at IS NULL", kandangID, startDate, endDate).
|
||||
Count(&conflictCount).Error; err != nil {
|
||||
if err := q.Count(&conflictCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if conflictCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrDateOverlapExist)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateNoEmptyKandangConflict(tx *gorm.DB, kandangID uint, startDate, endDate time.Time) error {
|
||||
func (s *dailyChecklistService) validateNoEmptyKandangRangeOverlap(tx *gorm.DB, kandangID uint, startDate, endDate time.Time, excludeDCID uint) error {
|
||||
q := tx.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("kandang_id = ? AND start_date <= ? AND end_date >= ?", kandangID, endDate, startDate)
|
||||
if excludeDCID > 0 {
|
||||
q = q.Where("daily_checklist_id <> ?", excludeDCID)
|
||||
}
|
||||
var overlapCount int64
|
||||
if err := q.Count(&overlapCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if overlapCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrEmptyKandangRangeOverlap)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateDateNotInEmptyKandangRange(tx *gorm.DB, kandangID uint, date time.Time, excludeDCID uint) error {
|
||||
q := tx.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("kandang_id = ? AND start_date <= ? AND end_date >= ?", kandangID, date, date)
|
||||
if excludeDCID > 0 {
|
||||
q = q.Where("daily_checklist_id <> ?", excludeDCID)
|
||||
}
|
||||
var rec entity.DailyChecklistEmptyKandang
|
||||
if err := q.First(&rec).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil
|
||||
}
|
||||
return err
|
||||
}
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrDateInsideEmptyKandang)
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateNoExistingEmptyKandangInRange(tx *gorm.DB, kandangID uint, startDate, endDate time.Time, excludeDCID uint) error {
|
||||
q := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("kandang_id = ? AND date BETWEEN ? AND ? AND category = ? AND deleted_at IS NULL",
|
||||
kandangID, startDate, endDate, dailyChecklistCategoryEmptyKandang)
|
||||
if excludeDCID > 0 {
|
||||
q = q.Where("id <> ?", excludeDCID)
|
||||
}
|
||||
var conflictCount int64
|
||||
if err := q.Count(&conflictCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if conflictCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrEmptyKandangRangeOverlap)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateDateNotInExistingEmptyKandangChecklist(tx *gorm.DB, kandangID uint, date time.Time, excludeDCID uint) error {
|
||||
q := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("kandang_id = ? AND date = ? AND category = ? AND deleted_at IS NULL",
|
||||
kandangID, date, dailyChecklistCategoryEmptyKandang)
|
||||
if excludeDCID > 0 {
|
||||
q = q.Where("id <> ?", excludeDCID)
|
||||
}
|
||||
var conflictCount int64
|
||||
if err := q.Count(&conflictCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if conflictCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrDateInsideEmptyKandang)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) upsertEmptyKandangRange(tx *gorm.DB, dailyChecklistID, kandangID uint, startDate, endDate time.Time, actorID uint) error {
|
||||
var existing entity.DailyChecklistEmptyKandang
|
||||
err := tx.Where("daily_checklist_id = ?", dailyChecklistID).First(&existing).Error
|
||||
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return err
|
||||
}
|
||||
|
||||
if err == nil {
|
||||
return tx.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("id = ?", existing.Id).
|
||||
Updates(map[string]any{
|
||||
"kandang_id": kandangID,
|
||||
"start_date": startDate,
|
||||
"end_date": endDate,
|
||||
"updated_at": time.Now(),
|
||||
}).Error
|
||||
}
|
||||
|
||||
record := &entity.DailyChecklistEmptyKandang{
|
||||
DailyChecklistId: dailyChecklistID,
|
||||
KandangId: kandangID,
|
||||
StartDate: startDate,
|
||||
EndDate: endDate,
|
||||
}
|
||||
if actorID > 0 {
|
||||
actor := actorID
|
||||
record.CreatedBy = &actor
|
||||
}
|
||||
return tx.Create(record).Error
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) validateNoDeletedNonEmptyKandangForDate(tx *gorm.DB, kandangID uint, date time.Time) error {
|
||||
var conflictCount int64
|
||||
if err := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("kandang_id = ? AND date BETWEEN ? AND ? AND category = ? AND deleted_at IS NULL", kandangID, startDate, endDate, dailyChecklistCategoryEmptyKandang).
|
||||
Unscoped().
|
||||
Where("kandang_id = ? AND date = ? AND deleted_at IS NULL AND category != ?", kandangID, date, dailyChecklistCategoryEmptyKandang).
|
||||
Count(&conflictCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if conflictCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, dailyChecklistErrEmptyKandangExist)
|
||||
return fiber.NewError(fiber.StatusBadRequest, dailyChecklistErrDeletedNonEmptyKandangExists)
|
||||
}
|
||||
|
||||
return nil
|
||||
@@ -856,6 +992,157 @@ func (s dailyChecklistService) BulkUpdate(c *fiber.Ctx, req *validation.BulkStat
|
||||
return updated, nil
|
||||
}
|
||||
|
||||
func (s *dailyChecklistService) UpdateByPut(c *fiber.Ctx, req *validation.Create, id uint) (*entity.DailyChecklist, error) {
|
||||
if err := s.Validate.Struct(req); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
if err := s.ensureChecklistAccess(c, id); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
date, err := time.Parse(dailyChecklistDateLayout, strings.TrimSpace(req.Date))
|
||||
if err != nil {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "invalid date format, use YYYY-MM-DD")
|
||||
}
|
||||
|
||||
category := req.Category
|
||||
if req.EmptyKandang {
|
||||
category = dailyChecklistCategoryEmptyKandang
|
||||
}
|
||||
|
||||
status := req.Status
|
||||
|
||||
var emptyEndDate time.Time
|
||||
if category == dailyChecklistCategoryEmptyKandang {
|
||||
trimmedEnd := strings.TrimSpace(req.EmptyKandangEndDate)
|
||||
if trimmedEnd == "" {
|
||||
emptyEndDate = date
|
||||
} else {
|
||||
parsedEnd, parseErr := time.Parse(dailyChecklistDateLayout, trimmedEnd)
|
||||
if parseErr != nil {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "invalid empty_kandang_end_date format, use YYYY-MM-DD")
|
||||
}
|
||||
if parsedEnd.Before(date) {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, dailyChecklistErrEmptyKandangEndDateInvalid)
|
||||
}
|
||||
emptyEndDate = parsedEnd
|
||||
}
|
||||
}
|
||||
|
||||
var wasBranchC bool // non-empty_kandang → empty_kandang transition
|
||||
err = s.Repository.DB().WithContext(c.Context()).Transaction(func(tx *gorm.DB) error {
|
||||
if err := s.lockKandangForChecklistCreation(tx, req.KandangId); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
var existing entity.DailyChecklist
|
||||
if err := tx.Where("id = ? AND deleted_at IS NULL", id).First(&existing).Error; err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return fiber.NewError(fiber.StatusNotFound, "DailyChecklist not found")
|
||||
}
|
||||
return err
|
||||
}
|
||||
|
||||
existingIsEmpty := existing.Category == dailyChecklistCategoryEmptyKandang
|
||||
newIsEmpty := category == dailyChecklistCategoryEmptyKandang
|
||||
|
||||
if newIsEmpty {
|
||||
if err := s.validateNoNormalChecklistInRange(tx, req.KandangId, date, emptyEndDate, id); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := s.validateNoEmptyKandangRangeOverlap(tx, req.KandangId, date, emptyEndDate, id); err != nil {
|
||||
return err
|
||||
}
|
||||
} else {
|
||||
if err := s.validateDateNotInEmptyKandangRange(tx, req.KandangId, date, id); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
var conflictCount int64
|
||||
if err := tx.Model(&entity.DailyChecklist{}).
|
||||
Where("id <> ? AND date = ? AND kandang_id = ? AND category = ? AND deleted_at IS NULL",
|
||||
id, date, req.KandangId, category).
|
||||
Count(&conflictCount).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if conflictCount > 0 {
|
||||
return fiber.NewError(fiber.StatusConflict, "DailyChecklist already exists with same date, kandang, and category")
|
||||
}
|
||||
|
||||
result := tx.Model(&entity.DailyChecklist{}).Where("id = ?", id).Updates(map[string]any{
|
||||
"date": date,
|
||||
"kandang_id": req.KandangId,
|
||||
"category": category,
|
||||
"status": status,
|
||||
})
|
||||
if result.Error != nil {
|
||||
return result.Error
|
||||
}
|
||||
if result.RowsAffected == 0 {
|
||||
return gorm.ErrRecordNotFound
|
||||
}
|
||||
|
||||
actorID, _ := m.ActorIDFromContext(c)
|
||||
if newIsEmpty {
|
||||
if err := s.upsertEmptyKandangRange(tx, id, req.KandangId, date, emptyEndDate, actorID); err != nil {
|
||||
return err
|
||||
}
|
||||
// Branch C: non-empty → empty_kandang, hard-delete task/progress data
|
||||
if !existingIsEmpty {
|
||||
wasBranchC = true
|
||||
if err := tx.Exec(`
|
||||
DELETE FROM daily_checklist_activity_task_assignments
|
||||
WHERE task_id IN (
|
||||
SELECT id FROM daily_checklist_activity_tasks WHERE checklist_id = ?
|
||||
)`, id).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if err := tx.Where("checklist_id = ?", id).Delete(&entity.DailyChecklistActivityTask{}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
if err := tx.Where("daily_checklist_id = ?", id).Delete(&entity.DailyChecklistTask{}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
} else if existingIsEmpty {
|
||||
updates := map[string]any{
|
||||
"deleted_at": time.Now(),
|
||||
}
|
||||
if actorID > 0 {
|
||||
updates["deleted_by"] = actorID
|
||||
}
|
||||
if err := tx.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("daily_checklist_id = ? AND deleted_at IS NULL", id).
|
||||
Updates(updates).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
// Branch C: delete DC documents outside transaction (storage is external)
|
||||
if wasBranchC && s.DocumentSvc != nil {
|
||||
docs, docErr := s.DocumentSvc.ListByTarget(c.Context(), string(utils.DocumentTypeDailyChecklist), uint64(id))
|
||||
if docErr == nil && len(docs) > 0 {
|
||||
docIDs := make([]uint, 0, len(docs))
|
||||
for _, doc := range docs {
|
||||
docIDs = append(docIDs, doc.Id)
|
||||
}
|
||||
if delErr := s.DocumentSvc.DeleteDocuments(c.Context(), docIDs, true); delErr != nil {
|
||||
s.Log.Errorf("Failed to delete documents for DC %d during empty_kandang conversion: %+v", id, delErr)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return s.GetOne(c, id)
|
||||
}
|
||||
|
||||
func (s dailyChecklistService) DeleteOne(c *fiber.Ctx, id uint) error {
|
||||
if err := s.ensureChecklistAccess(c, id); err != nil {
|
||||
return err
|
||||
@@ -887,6 +1174,15 @@ func (s dailyChecklistService) DeleteOne(c *fiber.Ctx, id uint) error {
|
||||
return gorm.ErrRecordNotFound
|
||||
}
|
||||
|
||||
if err := tx.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("daily_checklist_id = ? AND deleted_at IS NULL", id).
|
||||
Updates(map[string]any{
|
||||
"deleted_at": time.Now(),
|
||||
"deleted_by": actorID,
|
||||
}).Error; err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}); err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
@@ -1449,11 +1745,12 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
Group("a.id, a.name, loc.id, loc.name, k.id, k.name, e.id, e.name, p.id, p.name")
|
||||
}
|
||||
|
||||
var total int64
|
||||
// --- Count approved rows ---
|
||||
var approvedTotal int64
|
||||
groupedForCount := buildGroupedQuery()
|
||||
if err := s.Repository.DB().WithContext(c.Context()).
|
||||
Table("(?) AS grouped", groupedForCount).
|
||||
Count(&total).Error; err != nil {
|
||||
Count(&approvedTotal).Error; err != nil {
|
||||
s.Log.Errorf("Failed to count report data: %+v", err)
|
||||
return nil, 0, err
|
||||
}
|
||||
@@ -1473,19 +1770,197 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
TotalAssignments int64
|
||||
}
|
||||
|
||||
rows := make([]reportRow, 0)
|
||||
if err := buildGroupedQuery().
|
||||
Order("a.name, loc.name, k.name, e.name").
|
||||
Offset(offset).
|
||||
Limit(params.Limit).
|
||||
Scan(&rows).Error; err != nil {
|
||||
s.Log.Errorf("Failed to fetch report data: %+v", err)
|
||||
type fallbackRowType struct {
|
||||
AreaID uint
|
||||
AreaName string
|
||||
LocationID uint
|
||||
LocationName string
|
||||
KandangID uint
|
||||
KandangName string
|
||||
EmployeeID uint
|
||||
EmployeeName string
|
||||
}
|
||||
|
||||
// buildFallbackQ returns employees in kandangs that have NO approved checklist data
|
||||
// for the filtered period. Applies the same scope/area/location/kandang/employee filters.
|
||||
buildFallbackQ := func() *gorm.DB {
|
||||
approvedKandangSubQ := buildBase().Select("DISTINCT dc.kandang_id")
|
||||
q := s.Repository.DB().WithContext(c.Context()).
|
||||
Table("employee_kandangs ek").
|
||||
Joins("JOIN employees e ON e.id = ek.employee_id AND e.deleted_at IS NULL").
|
||||
Joins("JOIN kandang_groups k ON k.id = ek.kandang_id AND k.deleted_at IS NULL").
|
||||
Joins("JOIN locations loc ON loc.id = k.location_id AND loc.deleted_at IS NULL").
|
||||
Joins("JOIN areas a ON a.id = loc.area_id AND a.deleted_at IS NULL").
|
||||
Where("ek.kandang_id NOT IN (?)", approvedKandangSubQ).
|
||||
Select("e.id AS employee_id, e.name AS employee_name, k.id AS kandang_id, k.name AS kandang_name, loc.id AS location_id, loc.name AS location_name, a.id AS area_id, a.name AS area_name")
|
||||
q = m.ApplyScopeFilter(q, locationScope, "loc.id")
|
||||
q = m.ApplyScopeFilter(q, areaScope, "a.id")
|
||||
if params.AreaID != nil {
|
||||
q = q.Where("a.id = ?", *params.AreaID)
|
||||
}
|
||||
if params.LocationID != nil {
|
||||
q = q.Where("loc.id = ?", *params.LocationID)
|
||||
}
|
||||
if params.KandangID != nil {
|
||||
q = q.Where("ek.kandang_id = ?", *params.KandangID)
|
||||
}
|
||||
if params.EmployeeID != nil {
|
||||
q = q.Where("ek.employee_id = ?", *params.EmployeeID)
|
||||
}
|
||||
// PhaseID not applied: fallback rows have no phase data
|
||||
return q
|
||||
}
|
||||
|
||||
// --- Count fallback rows ---
|
||||
var fallbackTotal int64
|
||||
if err := s.Repository.DB().WithContext(c.Context()).
|
||||
Table("(?) AS fb", buildFallbackQ()).
|
||||
Count(&fallbackTotal).Error; err != nil {
|
||||
s.Log.Errorf("Failed to count fallback report data: %+v", err)
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
if len(rows) == 0 {
|
||||
total := approvedTotal + fallbackTotal
|
||||
|
||||
// --- Fetch ALL approved rows (pagination done in Go after merging with fallback) ---
|
||||
allApprovedRows := make([]reportRow, 0)
|
||||
if approvedTotal > 0 {
|
||||
if err := buildGroupedQuery().
|
||||
Order("a.name, loc.name, k.name, e.name").
|
||||
Scan(&allApprovedRows).Error; err != nil {
|
||||
s.Log.Errorf("Failed to fetch report data: %+v", err)
|
||||
return nil, 0, err
|
||||
}
|
||||
}
|
||||
|
||||
// --- Fetch ALL fallback rows ---
|
||||
allFallbackRows := make([]fallbackRowType, 0)
|
||||
if fallbackTotal > 0 {
|
||||
if err := buildFallbackQ().
|
||||
Order("a.name, loc.name, k.name, e.name").
|
||||
Scan(&allFallbackRows).Error; err != nil {
|
||||
s.Log.Errorf("Failed to fetch fallback report data: %+v", err)
|
||||
return nil, 0, err
|
||||
}
|
||||
}
|
||||
|
||||
// --- Merge approved + fallback and sort consistently ---
|
||||
type mergedEntry struct {
|
||||
AreaName string
|
||||
LocationName string
|
||||
KandangName string
|
||||
EmployeeName string
|
||||
IsApproved bool
|
||||
Idx int
|
||||
}
|
||||
|
||||
merged := make([]mergedEntry, 0, len(allApprovedRows)+len(allFallbackRows))
|
||||
for i, r := range allApprovedRows {
|
||||
merged = append(merged, mergedEntry{
|
||||
AreaName: r.AreaName, LocationName: r.LocationName,
|
||||
KandangName: r.KandangName, EmployeeName: r.EmployeeName,
|
||||
IsApproved: true, Idx: i,
|
||||
})
|
||||
}
|
||||
for i, r := range allFallbackRows {
|
||||
merged = append(merged, mergedEntry{
|
||||
AreaName: r.AreaName, LocationName: r.LocationName,
|
||||
KandangName: r.KandangName, EmployeeName: r.EmployeeName,
|
||||
IsApproved: false, Idx: i,
|
||||
})
|
||||
}
|
||||
sort.Slice(merged, func(i, j int) bool {
|
||||
a, b := merged[i], merged[j]
|
||||
if a.AreaName != b.AreaName {
|
||||
return a.AreaName < b.AreaName
|
||||
}
|
||||
if a.LocationName != b.LocationName {
|
||||
return a.LocationName < b.LocationName
|
||||
}
|
||||
if a.KandangName != b.KandangName {
|
||||
return a.KandangName < b.KandangName
|
||||
}
|
||||
return a.EmployeeName < b.EmployeeName
|
||||
})
|
||||
|
||||
// --- Apply Go-level pagination ---
|
||||
end := offset + params.Limit
|
||||
if end > len(merged) {
|
||||
end = len(merged)
|
||||
}
|
||||
if offset >= len(merged) {
|
||||
return []DailyChecklistReportItem{}, total, nil
|
||||
}
|
||||
pageData := merged[offset:end]
|
||||
|
||||
// --- Split page into approved vs fallback rows ---
|
||||
pageApproved := make([]reportRow, 0)
|
||||
pageFallback := make([]fallbackRowType, 0)
|
||||
for _, entry := range pageData {
|
||||
if entry.IsApproved {
|
||||
pageApproved = append(pageApproved, allApprovedRows[entry.Idx])
|
||||
} else {
|
||||
pageFallback = append(pageFallback, allFallbackRows[entry.Idx])
|
||||
}
|
||||
}
|
||||
|
||||
applyEmptyKandangFlags := func(items []DailyChecklistReportItem, kandangIDs []uint) error {
|
||||
if len(kandangIDs) == 0 {
|
||||
return nil
|
||||
}
|
||||
firstDay := time.Date(params.Year, time.Month(params.Month), 1, 0, 0, 0, 0, time.UTC)
|
||||
lastDay := firstDay.AddDate(0, 1, 0).AddDate(0, 0, -1)
|
||||
|
||||
type emptyRangeRec struct {
|
||||
KandangID uint
|
||||
StartDate time.Time
|
||||
EndDate time.Time
|
||||
}
|
||||
var rangeRecs []emptyRangeRec
|
||||
if err := s.Repository.DB().WithContext(c.Context()).
|
||||
Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("kandang_id IN ? AND start_date <= ? AND end_date >= ?",
|
||||
kandangIDs, lastDay, firstDay).
|
||||
Select("kandang_id, start_date, end_date").
|
||||
Scan(&rangeRecs).Error; err != nil {
|
||||
s.Log.Errorf("Failed to get empty kandang ranges for report: %+v", err)
|
||||
return err
|
||||
}
|
||||
|
||||
emptyDaysByKandang := make(map[uint]map[int]struct{})
|
||||
|
||||
for _, rec := range rangeRecs {
|
||||
effectiveStart := rec.StartDate
|
||||
if effectiveStart.Before(firstDay) {
|
||||
effectiveStart = firstDay
|
||||
}
|
||||
effectiveEnd := rec.EndDate
|
||||
if effectiveEnd.After(lastDay) {
|
||||
effectiveEnd = lastDay
|
||||
}
|
||||
if effectiveStart.After(effectiveEnd) {
|
||||
continue
|
||||
}
|
||||
|
||||
if _, ok := emptyDaysByKandang[rec.KandangID]; !ok {
|
||||
emptyDaysByKandang[rec.KandangID] = make(map[int]struct{})
|
||||
}
|
||||
for d := effectiveStart; !d.After(effectiveEnd); d = d.AddDate(0, 0, 1) {
|
||||
emptyDaysByKandang[rec.KandangID][d.Day()] = struct{}{}
|
||||
}
|
||||
}
|
||||
|
||||
for i, item := range items {
|
||||
daySet := emptyDaysByKandang[item.KandangID]
|
||||
for day := range daySet {
|
||||
key := strconv.Itoa(day)
|
||||
if _, exists := items[i].DailyActivities[key]; !exists {
|
||||
items[i].DailyActivities[key] = "Kandang kosong"
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
type comboKey struct {
|
||||
EmployeeID uint
|
||||
@@ -1507,7 +1982,7 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
kandangSet := make(map[uint]struct{})
|
||||
phaseSet := make(map[uint]struct{})
|
||||
|
||||
for _, row := range rows {
|
||||
for _, row := range pageApproved {
|
||||
key := comboKey{EmployeeID: row.EmployeeID, KandangID: row.KandangID, PhaseID: row.PhaseID}
|
||||
comboSet[key] = struct{}{}
|
||||
if _, ok := employeeSet[row.EmployeeID]; !ok {
|
||||
@@ -1648,8 +2123,9 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
return selected
|
||||
}
|
||||
|
||||
items := make([]DailyChecklistReportItem, len(rows))
|
||||
for i, row := range rows {
|
||||
// --- Build approved items (existing logic) ---
|
||||
approvedItems := make([]DailyChecklistReportItem, len(pageApproved))
|
||||
for i, row := range pageApproved {
|
||||
key := comboKey{EmployeeID: row.EmployeeID, KandangID: row.KandangID, PhaseID: row.PhaseID}
|
||||
|
||||
activities := dailyActivityMap[key]
|
||||
@@ -1659,7 +2135,7 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
|
||||
totalChecklist := 0
|
||||
categoryCounts := DailyChecklistReportCategory{}
|
||||
activityOutput := make(map[string]int, len(activities))
|
||||
activityOutput := make(map[string]any, len(activities))
|
||||
|
||||
for day, stat := range activities {
|
||||
activityOutput[day] = stat.Completed
|
||||
@@ -1696,7 +2172,7 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
kandangPercentage = int(math.Round(float64(kandangStat.Completed) / float64(kandangStat.Total) * 100))
|
||||
}
|
||||
|
||||
items[i] = DailyChecklistReportItem{
|
||||
approvedItems[i] = DailyChecklistReportItem{
|
||||
AreaID: row.AreaID,
|
||||
AreaName: row.AreaName,
|
||||
LocationID: row.LocationID,
|
||||
@@ -1717,5 +2193,55 @@ func (s dailyChecklistService) GetReport(c *fiber.Ctx, params *validation.Report
|
||||
}
|
||||
}
|
||||
|
||||
return items, total, nil
|
||||
// --- Build fallback items (kandangs with no approved data) ---
|
||||
fallbackItems := make([]DailyChecklistReportItem, len(pageFallback))
|
||||
for i, fb := range pageFallback {
|
||||
fallbackItems[i] = DailyChecklistReportItem{
|
||||
AreaID: fb.AreaID,
|
||||
AreaName: fb.AreaName,
|
||||
LocationID: fb.LocationID,
|
||||
LocationName: fb.LocationName,
|
||||
KandangID: fb.KandangID,
|
||||
KandangName: fb.KandangName,
|
||||
EmployeeID: fb.EmployeeID,
|
||||
EmployeeName: fb.EmployeeName,
|
||||
PhaseName: "",
|
||||
DailyActivities: map[string]any{},
|
||||
Summary: DailyChecklistReportSummary{},
|
||||
}
|
||||
}
|
||||
|
||||
// --- Reconstruct allItems in the sorted pageData order ---
|
||||
allItems := make([]DailyChecklistReportItem, len(pageData))
|
||||
approvedIdx := 0
|
||||
fallbackIdx := 0
|
||||
for i, entry := range pageData {
|
||||
if entry.IsApproved {
|
||||
allItems[i] = approvedItems[approvedIdx]
|
||||
approvedIdx++
|
||||
} else {
|
||||
allItems[i] = fallbackItems[fallbackIdx]
|
||||
fallbackIdx++
|
||||
}
|
||||
}
|
||||
|
||||
// --- Collect all kandangIDs on this page (approved + fallback) for empty_kandang flags ---
|
||||
allKandangSet := make(map[uint]struct{})
|
||||
for _, id := range kandangIDs {
|
||||
allKandangSet[id] = struct{}{}
|
||||
}
|
||||
for _, fb := range pageFallback {
|
||||
allKandangSet[fb.KandangID] = struct{}{}
|
||||
}
|
||||
allKandangIDs := make([]uint, 0, len(allKandangSet))
|
||||
for id := range allKandangSet {
|
||||
allKandangIDs = append(allKandangIDs, id)
|
||||
}
|
||||
|
||||
// --- Flag empty kandang days within this report month ---
|
||||
if err := applyEmptyKandangFlags(allItems, allKandangIDs); err != nil {
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
return allItems, total, nil
|
||||
}
|
||||
|
||||
@@ -208,8 +208,18 @@ func TestCreateOneAllowsBulkEmptyKandangWhenRangeHasOnlySoftDeletedChecklist(t *
|
||||
Count(&activeInRange).Error; err != nil {
|
||||
t.Fatalf("failed counting checklists in range: %v", err)
|
||||
}
|
||||
if activeInRange != 5 {
|
||||
t.Fatalf("expected 5 active checklists created for range, got %d", activeInRange)
|
||||
if activeInRange != 1 {
|
||||
t.Fatalf("expected 1 active empty_kandang checklist created for range, got %d", activeInRange)
|
||||
}
|
||||
|
||||
var emptyRangeCount int64
|
||||
if err := db.Model(&entity.DailyChecklistEmptyKandang{}).
|
||||
Where("kandang_id = ? AND start_date = ? AND end_date = ? AND deleted_at IS NULL", 1, mustDate(t, "2026-01-01"), mustDate(t, "2026-01-05")).
|
||||
Count(&emptyRangeCount).Error; err != nil {
|
||||
t.Fatalf("failed counting empty kandang ranges: %v", err)
|
||||
}
|
||||
if emptyRangeCount != 1 {
|
||||
t.Fatalf("expected 1 empty kandang range record for [2026-01-01, 2026-01-05], got %d", emptyRangeCount)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -304,6 +314,18 @@ func setupDailyChecklistServiceTest(t *testing.T) (DailyChecklistService, *gorm.
|
||||
updated_at DATETIME NULL,
|
||||
deleted_at DATETIME NULL
|
||||
)`,
|
||||
`CREATE TABLE daily_checklist_empty_kandangs (
|
||||
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
||||
daily_checklist_id INTEGER NOT NULL,
|
||||
kandang_id INTEGER NOT NULL,
|
||||
start_date DATE NOT NULL,
|
||||
end_date DATE NOT NULL,
|
||||
created_by INTEGER NULL,
|
||||
deleted_by INTEGER NULL,
|
||||
created_at DATETIME NULL,
|
||||
updated_at DATETIME NULL,
|
||||
deleted_at DATETIME NULL
|
||||
)`,
|
||||
`INSERT INTO areas (id, name, created_by, created_at, updated_at, deleted_at) VALUES (1, 'Area A', 1, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP, NULL)`,
|
||||
`INSERT INTO locations (id, name, address, area_id, created_by, created_at, updated_at, deleted_at) VALUES (1, 'Farm A', 'Address', 1, 1, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP, NULL)`,
|
||||
`INSERT INTO kandang_groups (id, name, status, location_id, pic_id, created_by, created_at, updated_at, deleted_at) VALUES (1, 'Kandang A', 'ACTIVE', 1, 1, 1, CURRENT_TIMESTAMP, CURRENT_TIMESTAMP, NULL)`,
|
||||
@@ -316,7 +338,8 @@ func setupDailyChecklistServiceTest(t *testing.T) (DailyChecklistService, *gorm.
|
||||
}
|
||||
|
||||
repo := repository.NewDailyChecklistRepository(db)
|
||||
svc := NewDailyChecklistService(repo, nil, validator.New(), nil)
|
||||
emptyRepo := repository.NewDailyChecklistEmptyKandangRepository(db)
|
||||
svc := NewDailyChecklistService(repo, emptyRepo, nil, validator.New(), nil)
|
||||
return svc, db
|
||||
}
|
||||
|
||||
|
||||
@@ -10,7 +10,7 @@ type Create struct {
|
||||
Category string `json:"category" validate:"required"`
|
||||
Status string `json:"status" validate:"required"`
|
||||
EmptyKandang bool `json:"empty_kandang"`
|
||||
EmptyKandangEndDate string `json:"empty_kandang_end_date"`
|
||||
EmptyKandangEndDate string `json:"empty_kandang_end_date" validate:"omitempty"`
|
||||
}
|
||||
|
||||
type Update struct {
|
||||
|
||||
@@ -119,9 +119,9 @@ func (r *DashboardRepositoryImpl) GetRecordingWeeklyMetrics(ctx context.Context,
|
||||
var rows []RecordingWeeklyMetric
|
||||
|
||||
weekExpr := `CASE
|
||||
WHEN r.day IS NULL OR r.day <= 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN ((r.day - 1) / 7 + 1) + 17
|
||||
ELSE ((r.day - 1) / 7 + 1)
|
||||
WHEN r.day IS NULL OR r.day < 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN (r.day / 7 + 1) + 17
|
||||
ELSE (r.day / 7 + 1)
|
||||
END`
|
||||
|
||||
db := r.DB().WithContext(ctx).
|
||||
@@ -503,9 +503,9 @@ func (r *DashboardRepositoryImpl) GetComparisonWeeklyMetrics(ctx context.Context
|
||||
|
||||
var rows []ComparisonWeeklyMetric
|
||||
weekExpr := `CASE
|
||||
WHEN r.day IS NULL OR r.day <= 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN ((r.day - 1) / 7 + 1) + 17
|
||||
ELSE ((r.day - 1) / 7 + 1)
|
||||
WHEN r.day IS NULL OR r.day < 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN (r.day / 7 + 1) + 17
|
||||
ELSE (r.day / 7 + 1)
|
||||
END`
|
||||
|
||||
db := r.DB().WithContext(ctx).
|
||||
@@ -574,9 +574,9 @@ func (r *DashboardRepositoryImpl) GetEggQualityWeeklyMetrics(ctx context.Context
|
||||
var rows []EggQualityWeeklyMetric
|
||||
|
||||
weekExpr := `CASE
|
||||
WHEN r.day IS NULL OR r.day <= 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN ((r.day - 1) / 7 + 1) + 17
|
||||
ELSE ((r.day - 1) / 7 + 1)
|
||||
WHEN r.day IS NULL OR r.day < 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN (r.day / 7 + 1) + 17
|
||||
ELSE (r.day / 7 + 1)
|
||||
END`
|
||||
|
||||
db := r.DB().WithContext(ctx).
|
||||
@@ -616,9 +616,9 @@ func (r *DashboardRepositoryImpl) GetEggWeightWeeklyGrams(ctx context.Context, s
|
||||
var rows []WeeklyEggWeightMetric
|
||||
|
||||
weekExpr := `CASE
|
||||
WHEN r.day IS NULL OR r.day <= 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN ((r.day - 1) / 7 + 1) + 17
|
||||
ELSE ((r.day - 1) / 7 + 1)
|
||||
WHEN r.day IS NULL OR r.day < 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN (r.day / 7 + 1) + 17
|
||||
ELSE (r.day / 7 + 1)
|
||||
END`
|
||||
|
||||
db := r.DB().WithContext(ctx).
|
||||
@@ -647,9 +647,9 @@ func (r *DashboardRepositoryImpl) GetFeedUsageWeeklyByUom(ctx context.Context, s
|
||||
var rows []WeeklyFeedUsageMetric
|
||||
|
||||
weekExpr := `CASE
|
||||
WHEN r.day IS NULL OR r.day <= 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN ((r.day - 1) / 7 + 1) + 17
|
||||
ELSE ((r.day - 1) / 7 + 1)
|
||||
WHEN r.day IS NULL OR r.day < 0 THEN 1
|
||||
WHEN UPPER(pf.category) = 'LAYING' THEN (r.day / 7 + 1) + 17
|
||||
ELSE (r.day / 7 + 1)
|
||||
END`
|
||||
|
||||
db := r.DB().WithContext(ctx).
|
||||
|
||||
@@ -65,6 +65,8 @@ func (u *ExpenseController) GetAll(c *fiber.Ctx) error {
|
||||
RealizationStatus: strings.TrimSpace(c.Query("realization_status", "")),
|
||||
ProjectFlockID: uint64(c.QueryInt("project_flock_id", 0)),
|
||||
ProjectFlockKandangID: uint64(c.QueryInt("project_flock_kandang_id", 0)),
|
||||
SortBy: strings.TrimSpace(c.Query("sort_by", "")),
|
||||
SortOrder: strings.TrimSpace(c.Query("sort_order", "")),
|
||||
}
|
||||
|
||||
if isAllExpenseExcelExportRequest(c) {
|
||||
@@ -481,6 +483,27 @@ func (u *ExpenseController) CompleteExpense(c *fiber.Ctx) error {
|
||||
})
|
||||
}
|
||||
|
||||
func (u *ExpenseController) Pay(c *fiber.Ctx) error {
|
||||
expenseID := c.Params("id")
|
||||
id, err := strconv.Atoi(expenseID)
|
||||
if err != nil {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid expense ID")
|
||||
}
|
||||
|
||||
expense, err := u.ExpenseService.Pay(c, uint(id))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return c.Status(fiber.StatusOK).
|
||||
JSON(response.Success{
|
||||
Code: fiber.StatusOK,
|
||||
Status: "success",
|
||||
Message: "Pay expense successfully",
|
||||
Data: expense,
|
||||
})
|
||||
}
|
||||
|
||||
func ensureExpenseBulkApprovalPermission(c *fiber.Ctx, targetStep approvalutils.ApprovalStep) error {
|
||||
requiredPerms := []string{}
|
||||
|
||||
|
||||
@@ -29,6 +29,7 @@ type ExpenseBaseDTO struct {
|
||||
RealizationDate *time.Time `json:"realization_date,omitempty"`
|
||||
TransactionDate time.Time `json:"transaction_date"`
|
||||
Location *locationDTO.LocationRelationDTO `json:"location,omitempty"`
|
||||
IsPaid bool `json:"is_paid"`
|
||||
}
|
||||
|
||||
type ExpenseListDTO struct {
|
||||
@@ -86,6 +87,7 @@ type KandangGroupDTO struct {
|
||||
type DocumentDTO struct {
|
||||
ID uint64 `json:"id"`
|
||||
Path string `json:"path"`
|
||||
Name string `json:"name"`
|
||||
}
|
||||
|
||||
// === MAPPERS ===
|
||||
@@ -126,6 +128,7 @@ func ToExpenseBaseDTO(e *entity.Expense) ExpenseBaseDTO {
|
||||
RealizationDate: realizationDate,
|
||||
TransactionDate: e.TransactionDate,
|
||||
Location: location,
|
||||
IsPaid: e.IsPaid,
|
||||
}
|
||||
}
|
||||
|
||||
@@ -184,6 +187,7 @@ func ToExpenseDetailDTO(e *entity.Expense) ExpenseDetailDTO {
|
||||
documents = append(documents, DocumentDTO{
|
||||
ID: uint64(doc.Id),
|
||||
Path: doc.Path,
|
||||
Name: doc.Name,
|
||||
})
|
||||
}
|
||||
|
||||
@@ -191,6 +195,7 @@ func ToExpenseDetailDTO(e *entity.Expense) ExpenseDetailDTO {
|
||||
realizationDocs = append(realizationDocs, DocumentDTO{
|
||||
ID: uint64(doc.Id),
|
||||
Path: doc.Path,
|
||||
Name: doc.Name,
|
||||
})
|
||||
}
|
||||
|
||||
|
||||
@@ -98,6 +98,7 @@ func (r *ExpenseRealizationRepositoryImpl) GetAllWithFilters(ctx context.Context
|
||||
return db.
|
||||
Preload("Expense").
|
||||
Preload("Expense.Supplier").
|
||||
Preload("Expense.Location").
|
||||
Preload("Kandang").
|
||||
Preload("Kandang.Location").
|
||||
Preload("Nonstock").
|
||||
@@ -177,10 +178,48 @@ func (r *ExpenseRealizationRepositoryImpl) GetAllWithFilters(ctx context.Context
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
sortExpr := "expense_realizations.created_at"
|
||||
order := "DESC"
|
||||
if filters.SortOrder == "asc" {
|
||||
order = "ASC"
|
||||
}
|
||||
switch filters.SortBy {
|
||||
case "po_number":
|
||||
sortExpr = "expenses.po_number"
|
||||
case "reference_number":
|
||||
sortExpr = "expenses.reference_number"
|
||||
case "realization_date":
|
||||
sortExpr = "expenses.realization_date"
|
||||
case "transaction_date":
|
||||
sortExpr = "expenses.transaction_date"
|
||||
case "category":
|
||||
sortExpr = "expenses.category"
|
||||
case "product":
|
||||
sortExpr = "(SELECT name FROM nonstocks WHERE id = expense_nonstocks.nonstock_id)"
|
||||
case "supplier":
|
||||
sortExpr = "suppliers.name"
|
||||
case "location":
|
||||
sortExpr = "(SELECT l.name FROM kandangs k JOIN locations l ON l.id = k.location_id WHERE k.id = expense_nonstocks.kandang_id)"
|
||||
case "kandang":
|
||||
sortExpr = "(SELECT name FROM kandangs WHERE id = expense_nonstocks.kandang_id)"
|
||||
case "qty_pengajuan":
|
||||
sortExpr = "expense_nonstocks.qty"
|
||||
case "price_pengajuan":
|
||||
sortExpr = "expense_nonstocks.price"
|
||||
case "total_pengajuan":
|
||||
sortExpr = "expense_nonstocks.qty * expense_nonstocks.price"
|
||||
case "qty_realisasi":
|
||||
sortExpr = "expense_realizations.qty"
|
||||
case "price_realisasi":
|
||||
sortExpr = "expense_realizations.price"
|
||||
case "total_realisasi":
|
||||
sortExpr = "expense_realizations.qty * expense_realizations.price"
|
||||
}
|
||||
|
||||
if err := db.
|
||||
Offset(offset).
|
||||
Limit(limit).
|
||||
Order("expense_realizations.created_at DESC").
|
||||
Order(sortExpr + " " + order).
|
||||
Find(&realizations).Error; err != nil {
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
@@ -36,6 +36,7 @@ func ExpenseRoutes(v1 fiber.Router, u user.UserService, s expense.ExpenseService
|
||||
route.Post("/:id/realizations", m.RequirePermissions(m.P_ExpenseCreateRealizations), ctrl.CreateRealization)
|
||||
route.Patch("/:id/realizations", m.RequirePermissions(m.P_ExpenseUpdateRealizations), ctrl.UpdateRealization)
|
||||
route.Post("/:id/complete", m.RequirePermissions(m.P_ExpenseCompleteExpense), ctrl.CompleteExpense)
|
||||
route.Patch("/:id/pay", m.RequirePermissions(m.P_ExpenseCreateRealizations), ctrl.Pay)
|
||||
route.Delete("/:id/documents/:documentId", m.RequirePermissions(m.P_ExpenseDocument), ctrl.DeleteDocument)
|
||||
route.Delete("/:id/realization-documents/:documentId", m.RequirePermissions(m.P_ExpenseDocumentRealizations), ctrl.DeleteRealizationDocument)
|
||||
}
|
||||
|
||||
@@ -36,6 +36,7 @@ type ExpenseService interface {
|
||||
DeleteOne(ctx *fiber.Ctx, id uint64) error
|
||||
CreateRealization(ctx *fiber.Ctx, expenseID uint, req *validation.CreateRealization) (*expenseDto.ExpenseDetailDTO, error)
|
||||
CompleteExpense(ctx *fiber.Ctx, id uint, notes *string) (*expenseDto.ExpenseDetailDTO, error)
|
||||
Pay(ctx *fiber.Ctx, id uint) (*expenseDto.ExpenseDetailDTO, error)
|
||||
UpdateRealization(ctx *fiber.Ctx, expenseID uint, req *validation.UpdateRealization) (*expenseDto.ExpenseDetailDTO, error)
|
||||
DeleteDocument(ctx *fiber.Ctx, expenseID uint, documentID uint64, isRealization bool) error
|
||||
Approval(ctx *fiber.Ctx, req *validation.ApprovalRequest, approvalType string) ([]expenseDto.ExpenseDetailDTO, error)
|
||||
@@ -288,7 +289,40 @@ func (s expenseService) GetAll(c *fiber.Ctx, params *validation.Query) ([]expens
|
||||
like,
|
||||
)
|
||||
}
|
||||
return db.Order("expenses.created_at DESC").Order("expenses.updated_at DESC")
|
||||
sortBy := strings.TrimSpace(params.SortBy)
|
||||
sortOrder := strings.ToUpper(strings.TrimSpace(params.SortOrder))
|
||||
if sortOrder == "" {
|
||||
sortOrder = "DESC"
|
||||
}
|
||||
|
||||
switch sortBy {
|
||||
case "reference_number":
|
||||
return db.Order("expenses.reference_number " + sortOrder)
|
||||
case "transaction_date":
|
||||
return db.Order("expenses.transaction_date " + sortOrder)
|
||||
case "realization_date":
|
||||
return db.Order("expenses.realization_date " + sortOrder)
|
||||
case "location":
|
||||
return db.Order("(SELECT COALESCE(name,'') FROM locations WHERE id = expenses.location_id) " + sortOrder)
|
||||
case "created_user":
|
||||
return db.Order("(SELECT COALESCE(name,'') FROM users WHERE id = expenses.created_by) " + sortOrder)
|
||||
case "supplier":
|
||||
return db.Order("(SELECT COALESCE(name,'') FROM suppliers WHERE id = expenses.supplier_id) " + sortOrder)
|
||||
case "grand_total":
|
||||
return db.Order(`(SELECT COALESCE(
|
||||
(SELECT SUM(er.qty * er.price) FROM expense_realizations er
|
||||
JOIN expense_nonstocks en ON en.id = er.expense_nonstock_id
|
||||
WHERE en.expense_id = expenses.id),
|
||||
(SELECT SUM(en2.qty * en2.price) FROM expense_nonstocks en2
|
||||
WHERE en2.expense_id = expenses.id),
|
||||
0)) ` + sortOrder)
|
||||
case "is_paid":
|
||||
return db.Order("expenses.is_paid " + sortOrder)
|
||||
case "created_at":
|
||||
return db.Order("expenses.created_at " + sortOrder)
|
||||
default:
|
||||
return db.Order("expenses.created_at DESC").Order("expenses.updated_at DESC")
|
||||
}
|
||||
})
|
||||
|
||||
if scopeErr != nil {
|
||||
@@ -342,6 +376,18 @@ func (s expenseService) GetOne(c *fiber.Ctx, id uint) (*expenseDto.ExpenseDetail
|
||||
expense.LatestApproval = approval
|
||||
|
||||
responseDTO := expenseDto.ToExpenseDetailDTO(expense)
|
||||
|
||||
for i := range responseDTO.Documents {
|
||||
if url, err := commonSvc.ResolveDocumentURL(c.Context(), s.DocumentSvc, responseDTO.Documents[i].Path, 15*time.Minute); err == nil && url != "" {
|
||||
responseDTO.Documents[i].Path = url
|
||||
}
|
||||
}
|
||||
for i := range responseDTO.RealizationDocs {
|
||||
if url, err := commonSvc.ResolveDocumentURL(c.Context(), s.DocumentSvc, responseDTO.RealizationDocs[i].Path, 15*time.Minute); err == nil && url != "" {
|
||||
responseDTO.RealizationDocs[i].Path = url
|
||||
}
|
||||
}
|
||||
|
||||
return &responseDTO, nil
|
||||
}
|
||||
|
||||
@@ -1298,6 +1344,41 @@ func (s *expenseService) CompleteExpense(c *fiber.Ctx, id uint, notes *string) (
|
||||
return responseDTO, nil
|
||||
}
|
||||
|
||||
func (s *expenseService) Pay(c *fiber.Ctx, id uint) (*expenseDto.ExpenseDetailDTO, error) {
|
||||
if err := commonSvc.EnsureRelations(c.Context(),
|
||||
commonSvc.RelationCheck{Name: "Expense", ID: &id, Exists: s.Repository.IdExists},
|
||||
); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
expense, err := s.Repository.GetByID(c.Context(), id, nil)
|
||||
if err != nil {
|
||||
return nil, fiber.NewError(fiber.StatusInternalServerError, "Failed to get expense")
|
||||
}
|
||||
if expense.IsPaid {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "Expense is already paid")
|
||||
}
|
||||
|
||||
latestApproval, err := s.ApprovalSvc.LatestByTarget(c.Context(), utils.ApprovalWorkflowExpense, id, func(db *gorm.DB) *gorm.DB {
|
||||
return db.Where("action = ?", entity.ApprovalActionApproved)
|
||||
})
|
||||
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return nil, fiber.NewError(fiber.StatusInternalServerError, "Failed to validate workflow")
|
||||
}
|
||||
if latestApproval == nil {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "Expense must be approved by Finance (step 4) before payment")
|
||||
}
|
||||
if latestApproval.StepNumber < uint16(utils.ExpenseStepFinance) {
|
||||
return nil, fiber.NewError(fiber.StatusBadRequest, "Expense must be approved by Finance (step 4) before payment")
|
||||
}
|
||||
|
||||
if err := s.Repository.PatchOne(c.Context(), id, map[string]any{"is_paid": true}, nil); err != nil {
|
||||
return nil, fiber.NewError(fiber.StatusInternalServerError, "Failed to update payment status")
|
||||
}
|
||||
|
||||
return s.GetOne(c, id)
|
||||
}
|
||||
|
||||
func (s *expenseService) UpdateRealization(c *fiber.Ctx, expenseID uint, req *validation.UpdateRealization) (*expenseDto.ExpenseDetailDTO, error) {
|
||||
if err := s.Validate.Struct(req); err != nil {
|
||||
|
||||
|
||||
@@ -54,6 +54,8 @@ type Query struct {
|
||||
RealizationStatus string `query:"realization_status" validate:"omitempty,max=100"`
|
||||
ProjectFlockID uint64 `query:"project_flock_id" validate:"omitempty,gt=0"`
|
||||
ProjectFlockKandangID uint64 `query:"project_flock_kandang_id" validate:"omitempty,gt=0"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,oneof=reference_number transaction_date realization_date location created_user supplier grand_total is_paid created_at"`
|
||||
SortOrder string `query:"sort_order" validate:"omitempty,oneof=asc desc ASC DESC"`
|
||||
}
|
||||
|
||||
type CreateRealization struct {
|
||||
|
||||
@@ -5,6 +5,7 @@ import (
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/modules/finance/transactions/dto"
|
||||
service "gitlab.com/mbugroup/lti-api.git/internal/modules/finance/transactions/services"
|
||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/finance/transactions/validations"
|
||||
@@ -13,6 +14,8 @@ import (
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
const transactionExcelExportFetchLimit = 99999999
|
||||
|
||||
type TransactionController struct {
|
||||
TransactionService service.TransactionService
|
||||
}
|
||||
@@ -97,6 +100,8 @@ func (u *TransactionController) GetAll(c *fiber.Ctx) error {
|
||||
CustomerIDs: customerIDs,
|
||||
SupplierIDs: supplierIDs,
|
||||
SortDate: c.Query("sort_date", ""),
|
||||
SortBy: c.Query("sort_by", ""),
|
||||
SortOrder: c.Query("sort_order", ""),
|
||||
StartDate: c.Query("start_date", ""),
|
||||
EndDate: c.Query("end_date", ""),
|
||||
}
|
||||
@@ -105,6 +110,14 @@ func (u *TransactionController) GetAll(c *fiber.Ctx) error {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "page and limit must be greater than 0")
|
||||
}
|
||||
|
||||
if isTransactionExcelExportRequest(c) {
|
||||
results, err := u.getAllTransactionsForExcel(c, query)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return exportTransactionListExcel(c, results)
|
||||
}
|
||||
|
||||
result, totalResults, err := u.TransactionService.GetAll(c, query)
|
||||
if err != nil {
|
||||
return err
|
||||
@@ -147,6 +160,32 @@ func (u *TransactionController) GetOne(c *fiber.Ctx) error {
|
||||
})
|
||||
}
|
||||
|
||||
func isTransactionExcelExportRequest(c *fiber.Ctx) bool {
|
||||
return strings.EqualFold(strings.TrimSpace(c.Query("export")), "excel")
|
||||
}
|
||||
|
||||
func (u *TransactionController) getAllTransactionsForExcel(c *fiber.Ctx, baseQuery *validation.Query) ([]entity.Payment, error) {
|
||||
query := *baseQuery
|
||||
query.Page = 1
|
||||
query.Limit = transactionExcelExportFetchLimit
|
||||
results := make([]entity.Payment, 0)
|
||||
for {
|
||||
pageResults, total, err := u.TransactionService.GetAll(c, &query)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if len(pageResults) == 0 || total == 0 {
|
||||
break
|
||||
}
|
||||
results = append(results, pageResults...)
|
||||
if int64(len(results)) >= total {
|
||||
break
|
||||
}
|
||||
query.Page++
|
||||
}
|
||||
return results, nil
|
||||
}
|
||||
|
||||
func (u *TransactionController) DeleteOne(c *fiber.Ctx) error {
|
||||
param := c.Params("id")
|
||||
|
||||
|
||||
@@ -0,0 +1,307 @@
|
||||
package controller
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"github.com/xuri/excelize/v2"
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
)
|
||||
|
||||
const transactionExportSheetName = "Transaksi"
|
||||
|
||||
func exportTransactionListExcel(c *fiber.Ctx, payments []entity.Payment) error {
|
||||
content, err := buildTransactionExportWorkbook(payments)
|
||||
if err != nil {
|
||||
return fiber.NewError(fiber.StatusInternalServerError, "failed to generate excel file")
|
||||
}
|
||||
|
||||
filename := fmt.Sprintf("transaksi_%s.xlsx", time.Now().Format("20060102_150405"))
|
||||
c.Set("Content-Type", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet")
|
||||
c.Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, filename))
|
||||
return c.Status(fiber.StatusOK).Send(content)
|
||||
}
|
||||
|
||||
func buildTransactionExportWorkbook(payments []entity.Payment) ([]byte, error) {
|
||||
file := excelize.NewFile()
|
||||
defer file.Close()
|
||||
|
||||
defaultSheet := file.GetSheetName(file.GetActiveSheetIndex())
|
||||
if defaultSheet != transactionExportSheetName {
|
||||
if err := file.SetSheetName(defaultSheet, transactionExportSheetName); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
if err := setTransactionExportColumns(file); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := setTransactionExportHeaders(file); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := setTransactionExportRows(file, payments); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
if err := file.SetPanes(transactionExportSheetName, &excelize.Panes{
|
||||
Freeze: true,
|
||||
YSplit: 1,
|
||||
TopLeftCell: "A2",
|
||||
ActivePane: "bottomLeft",
|
||||
}); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
buffer, err := file.WriteToBuffer()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return buffer.Bytes(), nil
|
||||
}
|
||||
|
||||
func setTransactionExportColumns(file *excelize.File) error {
|
||||
columnWidths := map[string]float64{
|
||||
"A": 20,
|
||||
"B": 22,
|
||||
"C": 18,
|
||||
"D": 25,
|
||||
"E": 14,
|
||||
"F": 16,
|
||||
"G": 16,
|
||||
"H": 22,
|
||||
"I": 22,
|
||||
"J": 18,
|
||||
"K": 18,
|
||||
"L": 18,
|
||||
"M": 30,
|
||||
"N": 22,
|
||||
"O": 20,
|
||||
}
|
||||
|
||||
sheet := transactionExportSheetName
|
||||
for col, width := range columnWidths {
|
||||
if err := file.SetColWidth(sheet, col, col, width); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return file.SetRowHeight(sheet, 1, 24)
|
||||
}
|
||||
|
||||
func setTransactionExportHeaders(file *excelize.File) error {
|
||||
sheet := transactionExportSheetName
|
||||
headers := []string{
|
||||
"Kode Pembayaran",
|
||||
"No. Referensi",
|
||||
"Tipe Transaksi",
|
||||
"Pihak",
|
||||
"Tipe Pihak",
|
||||
"Tanggal Bayar",
|
||||
"Metode Bayar",
|
||||
"Bank",
|
||||
"No. Rekening Bank",
|
||||
"Pemasukan",
|
||||
"Pengeluaran",
|
||||
"Nominal",
|
||||
"Catatan",
|
||||
"Dibuat Oleh",
|
||||
"Status",
|
||||
}
|
||||
|
||||
for i, header := range headers {
|
||||
colName, err := excelize.ColumnNumberToName(i + 1)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, colName+"1", header); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
headerStyle, err := file.NewStyle(&excelize.Style{
|
||||
Font: &excelize.Font{Bold: true, Color: "1F2937"},
|
||||
Fill: excelize.Fill{Type: "pattern", Pattern: 1, Color: []string{"DCEBFA"}},
|
||||
Alignment: &excelize.Alignment{Horizontal: "center", Vertical: "center"},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return file.SetCellStyle(sheet, "A1", "O1", headerStyle)
|
||||
}
|
||||
|
||||
func setTransactionExportRows(file *excelize.File, payments []entity.Payment) error {
|
||||
if len(payments) == 0 {
|
||||
return nil
|
||||
}
|
||||
|
||||
sheet := transactionExportSheetName
|
||||
for i, p := range payments {
|
||||
row := strconv.Itoa(i + 2)
|
||||
if err := writeTransactionExportRow(file, sheet, row, p); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
lastRow := strconv.Itoa(len(payments) + 1)
|
||||
|
||||
dataStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{Horizontal: "left", Vertical: "center", WrapText: true},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellStyle(sheet, "A2", "O"+lastRow, dataStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
numericStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{Horizontal: "right", Vertical: "center"},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return file.SetCellStyle(sheet, "J2", "L"+lastRow, numericStyle)
|
||||
}
|
||||
|
||||
func writeTransactionExportRow(file *excelize.File, sheet, row string, p entity.Payment) error {
|
||||
incomeAmount, expenseAmount := txAmounts(p.Direction, p.Nominal)
|
||||
|
||||
values := []interface{}{
|
||||
safeTxText(p.PaymentCode),
|
||||
safeTxRefNumber(p.ReferenceNumber),
|
||||
safeTxText(txTransactionType(p)),
|
||||
safeTxText(txPartyName(p)),
|
||||
safeTxText(p.PartyType),
|
||||
formatTxDate(p.PaymentDate),
|
||||
safeTxText(p.PaymentMethod),
|
||||
safeTxBank(p),
|
||||
safeTxBankAccount(p),
|
||||
incomeAmount,
|
||||
expenseAmount,
|
||||
p.Nominal,
|
||||
safeTxText(p.Notes),
|
||||
safeTxText(txCreatedBy(p)),
|
||||
formatTxStatus(p),
|
||||
}
|
||||
|
||||
for colIdx, val := range values {
|
||||
colName, err := excelize.ColumnNumberToName(colIdx + 1)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, colName+row, val); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func safeTxText(s string) string {
|
||||
trimmed := strings.TrimSpace(s)
|
||||
if trimmed == "" {
|
||||
return "-"
|
||||
}
|
||||
return trimmed
|
||||
}
|
||||
|
||||
func safeTxRefNumber(s *string) string {
|
||||
if s == nil {
|
||||
return "-"
|
||||
}
|
||||
return safeTxText(*s)
|
||||
}
|
||||
|
||||
func safeTxBank(p entity.Payment) string {
|
||||
if p.BankWarehouse.Id == 0 {
|
||||
return "-"
|
||||
}
|
||||
return safeTxText(p.BankWarehouse.Name)
|
||||
}
|
||||
|
||||
func safeTxBankAccount(p entity.Payment) string {
|
||||
if p.BankWarehouse.Id == 0 {
|
||||
return "-"
|
||||
}
|
||||
return safeTxText(p.BankWarehouse.AccountNumber)
|
||||
}
|
||||
|
||||
func formatTxDate(t time.Time) string {
|
||||
if t.IsZero() {
|
||||
return "-"
|
||||
}
|
||||
loc, err := time.LoadLocation("Asia/Jakarta")
|
||||
if err == nil {
|
||||
t = t.In(loc)
|
||||
}
|
||||
return t.Format("02-01-2006")
|
||||
}
|
||||
|
||||
func formatTxStatus(p entity.Payment) string {
|
||||
if p.LatestApproval == nil {
|
||||
return "-"
|
||||
}
|
||||
return safeTxText(p.LatestApproval.StepName)
|
||||
}
|
||||
|
||||
func txTransactionType(p entity.Payment) string {
|
||||
if p.TransactionType != "" {
|
||||
return p.TransactionType
|
||||
}
|
||||
return p.Direction
|
||||
}
|
||||
|
||||
func txPartyName(p entity.Payment) string {
|
||||
switch p.PartyType {
|
||||
case "CUSTOMER":
|
||||
if p.Customer != nil && p.Customer.Id != 0 {
|
||||
return p.Customer.Name
|
||||
}
|
||||
case "SUPPLIER":
|
||||
if p.Supplier != nil && p.Supplier.Id != 0 {
|
||||
return p.Supplier.Name
|
||||
}
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
func txCreatedBy(p entity.Payment) string {
|
||||
if p.CreatedUser.Id == 0 {
|
||||
return ""
|
||||
}
|
||||
return p.CreatedUser.Name
|
||||
}
|
||||
|
||||
func txAmounts(direction string, nominal float64) (income, expense float64) {
|
||||
switch strings.ToUpper(direction) {
|
||||
case "IN":
|
||||
return nominal, 0
|
||||
case "OUT":
|
||||
return 0, nominal
|
||||
default:
|
||||
return 0, 0
|
||||
}
|
||||
}
|
||||
@@ -72,27 +72,36 @@ func (s transactionService) GetAll(c *fiber.Ctx, params *validation.Query) ([]en
|
||||
transactions, total, err := s.Repository.GetAll(c.Context(), offset, params.Limit, func(db *gorm.DB) *gorm.DB {
|
||||
db = s.withRelations(db)
|
||||
|
||||
if params.Search != "" {
|
||||
like := "%" + strings.ToLower(strings.TrimSpace(params.Search)) + "%"
|
||||
needsPartyJoin := params.Search != "" || params.SortBy == "customer_name"
|
||||
needsBankJoin := params.Search != "" || params.SortBy == "bank"
|
||||
|
||||
if needsPartyJoin {
|
||||
db = db.Joins(
|
||||
"LEFT JOIN customers ON customers.id = payments.party_id AND payments.party_type = ? AND customers.deleted_at IS NULL",
|
||||
string(utils.PaymentPartyCustomer),
|
||||
).Joins(
|
||||
"LEFT JOIN suppliers ON suppliers.id = payments.party_id AND payments.party_type = ? AND suppliers.deleted_at IS NULL",
|
||||
string(utils.PaymentPartySupplier),
|
||||
).Joins(
|
||||
"LEFT JOIN banks ON banks.id = payments.bank_id AND banks.deleted_at IS NULL",
|
||||
)
|
||||
}
|
||||
if needsBankJoin {
|
||||
db = db.Joins("LEFT JOIN banks ON banks.id = payments.bank_id AND banks.deleted_at IS NULL")
|
||||
}
|
||||
|
||||
if params.Search != "" {
|
||||
like := "%" + strings.ToLower(strings.TrimSpace(params.Search)) + "%"
|
||||
db = db.Where(
|
||||
`LOWER(payment_code) LIKE ? OR
|
||||
`(LOWER(payment_code) LIKE ? OR
|
||||
LOWER(COALESCE(reference_number, '')) LIKE ? OR
|
||||
LOWER(COALESCE(payment_method, '')) LIKE ? OR
|
||||
LOWER(COALESCE(transaction_type, '')) LIKE ? OR
|
||||
LOWER(COALESCE(notes, '')) LIKE ? OR
|
||||
LOWER(COALESCE(customers.name, '')) LIKE ? OR
|
||||
LOWER(COALESCE(suppliers.name, '')) LIKE ? OR
|
||||
LOWER(COALESCE(banks.name, '')) LIKE ?`,
|
||||
like, like, like, like, like, like, like, like,
|
||||
LOWER(COALESCE(banks.name, '')) LIKE ? OR
|
||||
CAST(payments.nominal AS TEXT) LIKE ? OR
|
||||
TO_CHAR(payments.payment_date, 'YYYY-MM-DD') LIKE ?)`,
|
||||
like, like, like, like, like, like, like, like, like, like,
|
||||
)
|
||||
}
|
||||
|
||||
@@ -136,7 +145,7 @@ func (s transactionService) GetAll(c *fiber.Ctx, params *validation.Query) ([]en
|
||||
db = db.Where("payment_date < ?", *endDate)
|
||||
}
|
||||
|
||||
return applyTransactionSort(db, params.SortDate)
|
||||
return applyTransactionSort(db, params.SortBy, params.SortOrder, params.SortDate)
|
||||
})
|
||||
|
||||
if err != nil {
|
||||
@@ -268,13 +277,39 @@ func parseTransactionDateRange(startDate, endDate string) (*time.Time, *time.Tim
|
||||
return startPtr, endPtr, nil
|
||||
}
|
||||
|
||||
func applyTransactionSort(db *gorm.DB, sortDate string) *gorm.DB {
|
||||
func applyTransactionSort(db *gorm.DB, sortBy, sortOrder, sortDate string) *gorm.DB {
|
||||
order := "DESC"
|
||||
if strings.ToUpper(strings.TrimSpace(sortOrder)) == "ASC" {
|
||||
order = "ASC"
|
||||
}
|
||||
|
||||
switch strings.ToLower(strings.TrimSpace(sortBy)) {
|
||||
case "payment_code":
|
||||
return db.Order("payments.payment_code " + order)
|
||||
case "reference_number":
|
||||
return db.Order("payments.reference_number " + order)
|
||||
case "transaction_type":
|
||||
return db.Order("payments.transaction_type " + order)
|
||||
case "customer_name":
|
||||
return db.Order("COALESCE(customers.name, suppliers.name) " + order)
|
||||
case "payment_date":
|
||||
return db.Order("payments.payment_date " + order)
|
||||
case "created_at":
|
||||
return db.Order("payments.created_at " + order)
|
||||
case "payment_method":
|
||||
return db.Order("payments.payment_method " + order)
|
||||
case "bank":
|
||||
return db.Order("banks.account_number " + order)
|
||||
case "expense_amount":
|
||||
return db.Order("CASE WHEN payments.direction = 'OUT' THEN payments.nominal ELSE 0 END " + order)
|
||||
case "income_amount":
|
||||
return db.Order("CASE WHEN payments.direction = 'IN' THEN payments.nominal ELSE 0 END " + order)
|
||||
}
|
||||
|
||||
switch strings.ToLower(strings.TrimSpace(sortDate)) {
|
||||
case "created_at":
|
||||
return db.Order("created_at DESC").Order("payment_date DESC")
|
||||
case "payment_date":
|
||||
return db.Order("payment_date DESC").Order("created_at DESC")
|
||||
return db.Order("payments.created_at DESC").Order("payments.payment_date DESC")
|
||||
default:
|
||||
return db.Order("payment_date DESC").Order("created_at DESC")
|
||||
return db.Order("payments.payment_date DESC").Order("payments.created_at DESC")
|
||||
}
|
||||
}
|
||||
|
||||
@@ -10,13 +10,15 @@ type Update struct {
|
||||
|
||||
type Query struct {
|
||||
Page int `query:"page" validate:"omitempty,number,min=1,gt=0"`
|
||||
Limit int `query:"limit" validate:"omitempty,number,min=1,max=100,gt=0"`
|
||||
Limit int `query:"limit" validate:"omitempty,number,min=1,gt=0"`
|
||||
Search string `query:"search" validate:"omitempty,max=50"`
|
||||
TransactionTypes []string `query:"transaction_types" validate:"omitempty,dive,max=50"`
|
||||
BankIDs []uint `query:"bank_ids" validate:"omitempty,dive,gt=0"`
|
||||
CustomerIDs []uint `query:"customer_ids" validate:"omitempty,dive,gt=0"`
|
||||
SupplierIDs []uint `query:"supplier_ids" validate:"omitempty,dive,gt=0"`
|
||||
SortDate string `query:"sort_date" validate:"omitempty,oneof=created_at payment_date"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,oneof=payment_code reference_number transaction_type customer_name payment_date created_at payment_method bank expense_amount income_amount"`
|
||||
SortOrder string `query:"sort_order" validate:"omitempty,oneof=asc desc"`
|
||||
StartDate string `query:"start_date" validate:"omitempty,datetime=2006-01-02"`
|
||||
EndDate string `query:"end_date" validate:"omitempty,datetime=2006-01-02"`
|
||||
}
|
||||
|
||||
+9
-36
@@ -7,7 +7,6 @@ import (
|
||||
|
||||
"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"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
@@ -166,42 +165,16 @@ func (r *ProductWarehouseRepositoryImpl) ApplyFlagsFilter(db *gorm.DB, flags []s
|
||||
return db
|
||||
}
|
||||
|
||||
fallbackCategoryCodes := utils.LegacyProductCategoryCodesForFlags(flags)
|
||||
|
||||
db = db.
|
||||
Joins("JOIN products p_flag ON p_flag.id = product_warehouses.product_id").
|
||||
Joins("LEFT JOIN product_categories pc_flag ON pc_flag.id = p_flag.product_category_id")
|
||||
|
||||
actualFlagFilter := `
|
||||
EXISTS (
|
||||
SELECT 1
|
||||
FROM flags f_flag
|
||||
WHERE f_flag.flagable_id = p_flag.id
|
||||
AND f_flag.flagable_type = ?
|
||||
AND f_flag.name IN ?
|
||||
)
|
||||
`
|
||||
|
||||
if len(fallbackCategoryCodes) == 0 {
|
||||
return db.Where(actualFlagFilter, entity.FlagableTypeProduct, flags).Distinct()
|
||||
}
|
||||
|
||||
return db.
|
||||
Where(
|
||||
`(`+actualFlagFilter+`) OR (
|
||||
NOT EXISTS (
|
||||
SELECT 1
|
||||
FROM flags f_any
|
||||
WHERE f_any.flagable_id = p_flag.id
|
||||
AND f_any.flagable_type = ?
|
||||
)
|
||||
AND pc_flag.code IN ?
|
||||
)`,
|
||||
entity.FlagableTypeProduct,
|
||||
flags,
|
||||
entity.FlagableTypeProduct,
|
||||
fallbackCategoryCodes,
|
||||
).
|
||||
Where(`
|
||||
EXISTS (
|
||||
SELECT 1
|
||||
FROM flags f_flag
|
||||
WHERE f_flag.flagable_id = product_warehouses.product_id
|
||||
AND f_flag.flagable_type = ?
|
||||
AND f_flag.name IN ?
|
||||
)
|
||||
`, entity.FlagableTypeProduct, flags).
|
||||
Distinct()
|
||||
}
|
||||
|
||||
|
||||
+8
-5
@@ -117,7 +117,7 @@ func insertProductWarehouseTestFixtures(t *testing.T, db *gorm.DB) {
|
||||
}
|
||||
}
|
||||
|
||||
func TestApplyFlagsFilterIncludesLegacyCategoryFallback(t *testing.T) {
|
||||
func TestApplyFlagsFilterOnlyIncludesFlaggedProducts(t *testing.T) {
|
||||
db := setupProductWarehouseFlagFilterTestDB(t)
|
||||
repo := NewProductWarehouseRepository(db)
|
||||
ctx := context.Background()
|
||||
@@ -131,12 +131,14 @@ func TestApplyFlagsFilterIncludesLegacyCategoryFallback(t *testing.T) {
|
||||
t.Fatalf("unexpected error: %v", err)
|
||||
}
|
||||
|
||||
if len(ids) != 2 || ids[0] != 1 || ids[1] != 2 {
|
||||
t.Fatalf("expected flagged and legacy RAW rows to match, got %v", ids)
|
||||
// Only PW 1 (product 10, flagged PAKAN) should match.
|
||||
// PW 2 (product 20, no flags, RAW category) must not appear — legacy fallback removed.
|
||||
if len(ids) != 1 || ids[0] != 1 {
|
||||
t.Fatalf("expected only flagged row to match, got %v", ids)
|
||||
}
|
||||
}
|
||||
|
||||
func TestApplyFlagsFilterDoesNotFallbackWhenProductAlreadyHasDifferentFlags(t *testing.T) {
|
||||
func TestApplyFlagsFilterExcludesWrongFlaggedProducts(t *testing.T) {
|
||||
db := setupProductWarehouseFlagFilterTestDB(t)
|
||||
repo := NewProductWarehouseRepository(db)
|
||||
ctx := context.Background()
|
||||
@@ -150,8 +152,9 @@ func TestApplyFlagsFilterDoesNotFallbackWhenProductAlreadyHasDifferentFlags(t *t
|
||||
t.Fatalf("unexpected error: %v", err)
|
||||
}
|
||||
|
||||
// PW 3 belongs to an OVK-flagged product — must not appear when filtering for PAKAN.
|
||||
if len(ids) != 0 {
|
||||
t.Fatalf("expected OVK-flagged product not to match PAKAN fallback, got %v", ids)
|
||||
t.Fatalf("expected OVK-flagged product not to match PAKAN filter, got %v", ids)
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -10,6 +10,7 @@ import (
|
||||
adjustments "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/adjustments"
|
||||
productStocks "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-stocks"
|
||||
productWarehouses "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses"
|
||||
stockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs"
|
||||
transfers "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers"
|
||||
// MODULE IMPORTS
|
||||
)
|
||||
@@ -23,6 +24,7 @@ func RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Valida
|
||||
adjustments.AdjustmentModule{},
|
||||
transfers.TransferModule{},
|
||||
productStocks.ProductStockModule{},
|
||||
stockLogs.StockLogModule{},
|
||||
// MODULE REGISTRY
|
||||
}
|
||||
|
||||
|
||||
@@ -0,0 +1,66 @@
|
||||
package controller
|
||||
|
||||
import (
|
||||
"math"
|
||||
"strings"
|
||||
|
||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/validations"
|
||||
stockLogDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/dto"
|
||||
stockLogService "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/services"
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/response"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
type StockLogController struct {
|
||||
StockLogService stockLogService.StockLogService
|
||||
}
|
||||
|
||||
func NewStockLogController(s stockLogService.StockLogService) *StockLogController {
|
||||
return &StockLogController{
|
||||
StockLogService: s,
|
||||
}
|
||||
}
|
||||
|
||||
func (u *StockLogController) GetAll(c *fiber.Ctx) error {
|
||||
query := &validation.Query{
|
||||
Page: c.QueryInt("page", 1),
|
||||
Limit: c.QueryInt("limit", 10),
|
||||
ProductWarehouseID: uint(c.QueryInt("product_warehouse_id", 0)),
|
||||
}
|
||||
|
||||
if query.Page < 1 || query.Limit < 1 {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "page and limit must be greater than 0")
|
||||
}
|
||||
|
||||
// Export to Excel
|
||||
if strings.EqualFold(strings.TrimSpace(c.Query("export")), "excel") {
|
||||
if query.ProductWarehouseID == 0 {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "product_warehouse_id is required for export")
|
||||
}
|
||||
results, err := u.StockLogService.GetAllForExport(c, query.ProductWarehouseID)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return exportStockLogListExcel(c, results)
|
||||
}
|
||||
|
||||
result, totalResults, err := u.StockLogService.GetAll(c, query)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return c.Status(fiber.StatusOK).
|
||||
JSON(response.SuccessWithPaginate[stockLogDTO.StockLogListDTO]{
|
||||
Code: fiber.StatusOK,
|
||||
Status: "success",
|
||||
Message: "Get all stock logs successfully",
|
||||
Meta: response.Meta{
|
||||
Page: query.Page,
|
||||
Limit: query.Limit,
|
||||
TotalPages: int64(math.Ceil(float64(totalResults) / float64(query.Limit))),
|
||||
TotalResults: totalResults,
|
||||
},
|
||||
Data: stockLogDTO.ToStockLogListDTOs(result),
|
||||
})
|
||||
}
|
||||
@@ -0,0 +1,118 @@
|
||||
package controller
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"time"
|
||||
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"github.com/xuri/excelize/v2"
|
||||
)
|
||||
|
||||
func exportStockLogListExcel(c *fiber.Ctx, stockLogs []entity.StockLog) error {
|
||||
file := excelize.NewFile()
|
||||
defer file.Close()
|
||||
|
||||
sheet := "Stock Logs"
|
||||
file.SetSheetName("Sheet1", sheet)
|
||||
|
||||
headers := []string{
|
||||
"ID",
|
||||
"Tanggal",
|
||||
"Gudang",
|
||||
"Stok Akhir",
|
||||
"Peningkatan",
|
||||
"Penurunan",
|
||||
"Jenis Transaksi",
|
||||
"Catatan",
|
||||
"Oleh",
|
||||
}
|
||||
|
||||
// Column widths
|
||||
colWidths := map[string]float64{
|
||||
"A": 8,
|
||||
"B": 20,
|
||||
"C": 25,
|
||||
"D": 14,
|
||||
"E": 14,
|
||||
"F": 14,
|
||||
"G": 20,
|
||||
"H": 30,
|
||||
"I": 20,
|
||||
}
|
||||
for col, width := range colWidths {
|
||||
file.SetColWidth(sheet, col, col, width)
|
||||
}
|
||||
|
||||
// Header style
|
||||
headerStyle, _ := file.NewStyle(&excelize.Style{
|
||||
Font: &excelize.Font{
|
||||
Bold: true,
|
||||
Size: 11,
|
||||
},
|
||||
Fill: excelize.Fill{
|
||||
Type: "pattern",
|
||||
Pattern: 1,
|
||||
Color: []string{"D9E1F2"},
|
||||
},
|
||||
Border: []excelize.Border{
|
||||
{Type: "bottom", Style: 1, Color: "000000"},
|
||||
},
|
||||
})
|
||||
|
||||
// Write header row
|
||||
for i, h := range headers {
|
||||
cell, _ := excelize.CoordinatesToCellName(i+1, 1)
|
||||
file.SetCellValue(sheet, cell, h)
|
||||
file.SetCellStyle(sheet, cell, cell, headerStyle)
|
||||
}
|
||||
|
||||
// Freeze header row
|
||||
file.SetPanes(sheet, &excelize.Panes{
|
||||
Freeze: true,
|
||||
YSplit: 1,
|
||||
TopLeftCell: "A2",
|
||||
ActivePane: "bottomLeft",
|
||||
})
|
||||
|
||||
// Write data rows
|
||||
for i, log := range stockLogs {
|
||||
row := i + 2
|
||||
|
||||
warehouseName := ""
|
||||
if log.ProductWarehouse != nil {
|
||||
warehouseName = log.ProductWarehouse.Warehouse.Name
|
||||
}
|
||||
|
||||
userName := ""
|
||||
if log.CreatedUser != nil {
|
||||
userName = log.CreatedUser.Name
|
||||
}
|
||||
|
||||
notes := ""
|
||||
if log.Notes != "" {
|
||||
notes = log.Notes
|
||||
}
|
||||
|
||||
file.SetCellInt(sheet, fmt.Sprintf("A%d", row), int(log.Id))
|
||||
file.SetCellValue(sheet, fmt.Sprintf("B%d", row), log.CreatedAt.Format("2006-01-02 15:04:05"))
|
||||
file.SetCellValue(sheet, fmt.Sprintf("C%d", row), warehouseName)
|
||||
file.SetCellFloat(sheet, fmt.Sprintf("D%d", row), log.Stock, 3, 64)
|
||||
file.SetCellFloat(sheet, fmt.Sprintf("E%d", row), log.Increase, 3, 64)
|
||||
file.SetCellFloat(sheet, fmt.Sprintf("F%d", row), log.Decrease, 3, 64)
|
||||
file.SetCellValue(sheet, fmt.Sprintf("G%d", row), log.LoggableType)
|
||||
file.SetCellValue(sheet, fmt.Sprintf("H%d", row), notes)
|
||||
file.SetCellValue(sheet, fmt.Sprintf("I%d", row), userName)
|
||||
}
|
||||
|
||||
buffer, err := file.WriteToBuffer()
|
||||
if err != nil {
|
||||
return fiber.NewError(fiber.StatusInternalServerError, "failed to generate excel file")
|
||||
}
|
||||
|
||||
filename := fmt.Sprintf("stock_logs_%s.xlsx", time.Now().Format("20060102_150405"))
|
||||
c.Set("Content-Type", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet")
|
||||
c.Set("Content-Disposition", fmt.Sprintf(`attachment; filename="%s"`, filename))
|
||||
return c.Status(fiber.StatusOK).Send(buffer.Bytes())
|
||||
}
|
||||
@@ -0,0 +1,61 @@
|
||||
package dto
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
userDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/users/dto"
|
||||
)
|
||||
|
||||
type StockLogListDTO struct {
|
||||
Id uint `json:"id"`
|
||||
ProductWarehouseId uint `json:"product_warehouse_id"`
|
||||
Increase float64 `json:"increase"`
|
||||
Decrease float64 `json:"decrease"`
|
||||
Stock float64 `json:"stock"`
|
||||
LoggableType string `json:"loggable_type"`
|
||||
LoggableId uint `json:"loggable_id"`
|
||||
Notes *string `json:"notes"`
|
||||
CreatedBy uint `json:"created_by"`
|
||||
CreatedUser *userDTO.UserRelationDTO `json:"created_user,omitempty"`
|
||||
CreatedAt time.Time `json:"created_at"`
|
||||
}
|
||||
|
||||
func ToStockLogListDTO(e entity.StockLog) StockLogListDTO {
|
||||
var notes *string
|
||||
if e.Notes != "" {
|
||||
n := e.Notes
|
||||
notes = &n
|
||||
}
|
||||
|
||||
var createdUser *userDTO.UserRelationDTO
|
||||
if e.CreatedUser != nil && e.CreatedUser.Id != 0 {
|
||||
mapped := userDTO.ToUserRelationDTO(*e.CreatedUser)
|
||||
createdUser = &mapped
|
||||
}
|
||||
|
||||
return StockLogListDTO{
|
||||
Id: e.Id,
|
||||
ProductWarehouseId: e.ProductWarehouseId,
|
||||
Increase: e.Increase,
|
||||
Decrease: e.Decrease,
|
||||
Stock: e.Stock,
|
||||
LoggableType: e.LoggableType,
|
||||
LoggableId: e.LoggableId,
|
||||
Notes: notes,
|
||||
CreatedBy: e.CreatedBy,
|
||||
CreatedUser: createdUser,
|
||||
CreatedAt: e.CreatedAt,
|
||||
}
|
||||
}
|
||||
|
||||
func ToStockLogListDTOs(e []entity.StockLog) []StockLogListDTO {
|
||||
if len(e) == 0 {
|
||||
return []StockLogListDTO{}
|
||||
}
|
||||
result := make([]StockLogListDTO, len(e))
|
||||
for i, log := range e {
|
||||
result[i] = ToStockLogListDTO(log)
|
||||
}
|
||||
return result
|
||||
}
|
||||
@@ -0,0 +1,24 @@
|
||||
package stockLogs
|
||||
|
||||
import (
|
||||
"github.com/go-playground/validator/v10"
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"gorm.io/gorm"
|
||||
|
||||
stockLogService "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/services"
|
||||
stockLogRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
||||
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
||||
)
|
||||
|
||||
type StockLogModule struct{}
|
||||
|
||||
func (StockLogModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
||||
userRepo := rUser.NewUserRepository(db)
|
||||
userService := sUser.NewUserService(userRepo, validate)
|
||||
|
||||
stockLogRepo := stockLogRepo.NewStockLogRepository(db)
|
||||
service := stockLogService.NewStockLogService(stockLogRepo, validate)
|
||||
|
||||
StockLogRoutes(router, userService, service)
|
||||
}
|
||||
@@ -0,0 +1,19 @@
|
||||
package stockLogs
|
||||
|
||||
import (
|
||||
m "gitlab.com/mbugroup/lti-api.git/internal/middleware"
|
||||
controller "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/controllers"
|
||||
stockLog "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/services"
|
||||
user "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
||||
|
||||
"github.com/gofiber/fiber/v2"
|
||||
)
|
||||
|
||||
func StockLogRoutes(v1 fiber.Router, u user.UserService, s stockLog.StockLogService) {
|
||||
ctrl := controller.NewStockLogController(s)
|
||||
|
||||
route := v1.Group("/stock-logs")
|
||||
route.Use(m.Auth(u))
|
||||
|
||||
route.Get("/", m.RequirePermissions(m.P_StockLogGetAll), ctrl.GetAll)
|
||||
}
|
||||
@@ -0,0 +1,125 @@
|
||||
package service
|
||||
|
||||
import (
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
m "gitlab.com/mbugroup/lti-api.git/internal/middleware"
|
||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/stock-logs/validations"
|
||||
stockLogRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||
|
||||
"github.com/go-playground/validator/v10"
|
||||
"github.com/gofiber/fiber/v2"
|
||||
"github.com/sirupsen/logrus"
|
||||
"gorm.io/gorm"
|
||||
)
|
||||
|
||||
type StockLogService interface {
|
||||
GetAll(ctx *fiber.Ctx, params *validation.Query) ([]entity.StockLog, int64, error)
|
||||
GetAllForExport(ctx *fiber.Ctx, productWarehouseID uint) ([]entity.StockLog, error)
|
||||
}
|
||||
|
||||
type stockLogService struct {
|
||||
Log *logrus.Logger
|
||||
Validate *validator.Validate
|
||||
StockLogRepo stockLogRepo.StockLogRepository
|
||||
}
|
||||
|
||||
func NewStockLogService(
|
||||
stockLogRepo stockLogRepo.StockLogRepository,
|
||||
validate *validator.Validate,
|
||||
) StockLogService {
|
||||
return &stockLogService{
|
||||
Log: utils.Log,
|
||||
Validate: validate,
|
||||
StockLogRepo: stockLogRepo,
|
||||
}
|
||||
}
|
||||
|
||||
func (s *stockLogService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.StockLog, int64, error) {
|
||||
if err := s.Validate.Struct(params); err != nil {
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
locationScope, areaScope, err := m.ResolveLocationAreaScopes(c, s.StockLogRepo.DB())
|
||||
if err != nil {
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
offset := (params.Page - 1) * params.Limit
|
||||
|
||||
stockLogs, total, err := s.StockLogRepo.GetAll(c.Context(), offset, params.Limit, func(db *gorm.DB) *gorm.DB {
|
||||
db = db.Where("product_warehouse_id = ?", params.ProductWarehouseID)
|
||||
|
||||
if locationScope.Restrict || areaScope.Restrict {
|
||||
if (locationScope.Restrict && len(locationScope.IDs) == 0) || (areaScope.Restrict && len(areaScope.IDs) == 0) {
|
||||
return db.Where("1 = 0")
|
||||
}
|
||||
db = db.
|
||||
Joins("JOIN product_warehouses pw ON pw.id = stock_logs.product_warehouse_id").
|
||||
Joins("JOIN warehouses w ON w.id = pw.warehouse_id")
|
||||
if locationScope.Restrict {
|
||||
db = db.Where("w.location_id IN ?", locationScope.IDs)
|
||||
}
|
||||
if areaScope.Restrict {
|
||||
db = db.Where("w.area_id IN ?", areaScope.IDs)
|
||||
}
|
||||
}
|
||||
|
||||
db = db.
|
||||
Preload("CreatedUser").
|
||||
Order("stock_logs.created_at DESC")
|
||||
|
||||
return db
|
||||
})
|
||||
if err != nil {
|
||||
s.Log.Errorf("Failed to get stock logs: %+v", err)
|
||||
return nil, 0, err
|
||||
}
|
||||
|
||||
if total == 0 {
|
||||
return []entity.StockLog{}, 0, nil
|
||||
}
|
||||
|
||||
return stockLogs, total, nil
|
||||
}
|
||||
|
||||
func (s *stockLogService) GetAllForExport(c *fiber.Ctx, productWarehouseID uint) ([]entity.StockLog, error) {
|
||||
locationScope, areaScope, err := m.ResolveLocationAreaScopes(c, s.StockLogRepo.DB())
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
stockLogs, _, err := s.StockLogRepo.GetAll(c.Context(), 0, -1, func(db *gorm.DB) *gorm.DB {
|
||||
db = db.Where("product_warehouse_id = ?", productWarehouseID)
|
||||
|
||||
if locationScope.Restrict || areaScope.Restrict {
|
||||
if (locationScope.Restrict && len(locationScope.IDs) == 0) || (areaScope.Restrict && len(areaScope.IDs) == 0) {
|
||||
return db.Where("1 = 0")
|
||||
}
|
||||
db = db.
|
||||
Joins("JOIN product_warehouses pw ON pw.id = stock_logs.product_warehouse_id").
|
||||
Joins("JOIN warehouses w ON w.id = pw.warehouse_id")
|
||||
if locationScope.Restrict {
|
||||
db = db.Where("w.location_id IN ?", locationScope.IDs)
|
||||
}
|
||||
if areaScope.Restrict {
|
||||
db = db.Where("w.area_id IN ?", areaScope.IDs)
|
||||
}
|
||||
}
|
||||
|
||||
db = db.
|
||||
Preload("CreatedUser").
|
||||
Preload("ProductWarehouse").
|
||||
Preload("ProductWarehouse.Warehouse").
|
||||
Order("stock_logs.created_at ASC")
|
||||
|
||||
return db
|
||||
})
|
||||
if err != nil {
|
||||
s.Log.Errorf("Failed to get stock logs for export: %+v", err)
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return stockLogs, nil
|
||||
}
|
||||
|
||||
@@ -0,0 +1,7 @@
|
||||
package validation
|
||||
|
||||
type Query struct {
|
||||
Page int `query:"page" validate:"omitempty,number,min=1,gt=0"`
|
||||
Limit int `query:"limit" validate:"omitempty,number,min=1,gt=0"`
|
||||
ProductWarehouseID uint `query:"product_warehouse_id" validate:"required,gt=0"`
|
||||
}
|
||||
@@ -56,6 +56,12 @@ func (u *DeliveryOrdersController) GetAll(c *fiber.Ctx) error {
|
||||
return fiber.NewError(fiber.StatusBadRequest, "Invalid product_ids")
|
||||
}
|
||||
|
||||
sortBy := strings.TrimSpace(c.Query("sort_by", ""))
|
||||
sortOrder := strings.TrimSpace(c.Query("sort_order", ""))
|
||||
if sortOrder == "" {
|
||||
sortOrder = "asc"
|
||||
}
|
||||
|
||||
query := &validation.DeliveryOrderQuery{
|
||||
Page: c.QueryInt("page", 1),
|
||||
Limit: c.QueryInt("limit", 10),
|
||||
@@ -66,6 +72,8 @@ func (u *DeliveryOrdersController) GetAll(c *fiber.Ctx) error {
|
||||
MarketingId: uint(c.QueryInt("marketing_id", 0)),
|
||||
ProjectFlockID: uint(c.QueryInt("project_flock_id", 0)),
|
||||
ProjectFlockKandangID: uint(c.QueryInt("project_flock_kandang_id", 0)),
|
||||
SortBy: sortBy,
|
||||
SortOrder: sortOrder,
|
||||
}
|
||||
|
||||
if isAllExcelExportRequest(c) {
|
||||
|
||||
@@ -2,7 +2,6 @@ package controller
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"math"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
@@ -76,9 +75,18 @@ func setMarketingExportColumns(file *excelize.File, sheet string) error {
|
||||
"B": 14,
|
||||
"C": 18,
|
||||
"D": 20,
|
||||
"E": 18,
|
||||
"F": 60,
|
||||
"G": 24,
|
||||
"E": 14,
|
||||
"F": 40,
|
||||
"G": 10,
|
||||
"H": 12,
|
||||
"I": 12,
|
||||
"J": 12,
|
||||
"K": 16,
|
||||
"L": 16,
|
||||
"M": 18,
|
||||
"N": 18,
|
||||
"O": 18,
|
||||
"P": 24,
|
||||
}
|
||||
|
||||
for col, width := range columnWidths {
|
||||
@@ -96,13 +104,22 @@ func setMarketingExportColumns(file *excelize.File, sheet string) error {
|
||||
|
||||
func setMarketingExportHeaders(file *excelize.File, sheet string) error {
|
||||
headers := []string{
|
||||
"No. Order",
|
||||
"Tanggal",
|
||||
"Status",
|
||||
"Customer",
|
||||
"Grand Total",
|
||||
"Products",
|
||||
"Notes",
|
||||
"No. Order", // A
|
||||
"Tanggal", // B
|
||||
"Status", // C
|
||||
"Customer", // D
|
||||
"Tipe", // E
|
||||
"Nama Produk", // F
|
||||
"Week", // G
|
||||
"Jumlah", // H
|
||||
"Satuan", // I
|
||||
"Qty Peti", // J
|
||||
"Berat Rata-rata (kg)", // K
|
||||
"Total Berat (kg)", // L
|
||||
"Harga Satuan", // M
|
||||
"Total Harga", // N
|
||||
"Grand Total", // O
|
||||
"Catatan", // P
|
||||
}
|
||||
|
||||
for i, header := range headers {
|
||||
@@ -131,7 +148,7 @@ func setMarketingExportHeaders(file *excelize.File, sheet string) error {
|
||||
return err
|
||||
}
|
||||
|
||||
return file.SetCellStyle(sheet, "A1", "G1", headerStyle)
|
||||
return file.SetCellStyle(sheet, "A1", "P1", headerStyle)
|
||||
}
|
||||
|
||||
func setMarketingExportRows(file *excelize.File, sheet string, items []dto.MarketingListDTO) error {
|
||||
@@ -139,70 +156,154 @@ func setMarketingExportRows(file *excelize.File, sheet string, items []dto.Marke
|
||||
return nil
|
||||
}
|
||||
|
||||
for i, item := range items {
|
||||
rowNumber := i + 2
|
||||
if err := file.SetCellValue(sheet, "A"+strconv.Itoa(rowNumber), safeMarketingExportText(item.SoNumber)); err != nil {
|
||||
return err
|
||||
row := 1
|
||||
for _, item := range items {
|
||||
soNumber := safeMarketingExportText(item.SoNumber)
|
||||
soDate := formatMarketingExportDate(item.SoDate)
|
||||
status := formatMarketingExportStatus(item)
|
||||
customer := safeMarketingExportText(item.Customer.Name)
|
||||
grandTotal := sumMarketingGrandTotal(item.SalesOrder)
|
||||
notes := safeMarketingExportText(item.Notes)
|
||||
|
||||
if len(item.SalesOrder) == 0 {
|
||||
row++
|
||||
r := strconv.Itoa(row)
|
||||
vals := map[string]interface{}{
|
||||
"A": soNumber, "B": soDate, "C": status, "D": customer,
|
||||
"E": "-", "F": "-", "G": "-", "H": "-", "I": "-", "J": "-",
|
||||
"K": "-", "L": "-", "M": "-", "N": "-",
|
||||
"O": grandTotal, "P": notes,
|
||||
}
|
||||
for col, val := range vals {
|
||||
if err := file.SetCellValue(sheet, col+r, val); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
continue
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "B"+strconv.Itoa(rowNumber), formatMarketingExportDate(item.SoDate)); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "C"+strconv.Itoa(rowNumber), formatMarketingExportStatus(item)); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "D"+strconv.Itoa(rowNumber), safeMarketingExportText(item.Customer.Name)); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "E"+strconv.Itoa(rowNumber), formatMarketingRupiah(sumMarketingGrandTotal(item.SalesOrder))); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "F"+strconv.Itoa(rowNumber), formatMarketingProducts(item.SalesOrder)); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "G"+strconv.Itoa(rowNumber), safeMarketingExportText(item.Notes)); err != nil {
|
||||
return err
|
||||
|
||||
for _, prod := range item.SalesOrder {
|
||||
row++
|
||||
r := strconv.Itoa(row)
|
||||
|
||||
productName := "-"
|
||||
if prod.ProductWarehouse != nil && prod.ProductWarehouse.Product != nil {
|
||||
if n := strings.TrimSpace(prod.ProductWarehouse.Product.Name); n != "" {
|
||||
productName = n
|
||||
}
|
||||
}
|
||||
|
||||
week := "-"
|
||||
if prod.Week != nil {
|
||||
week = strconv.Itoa(*prod.Week)
|
||||
}
|
||||
|
||||
satuan := "-"
|
||||
if prod.ConvertionUnit != nil && strings.TrimSpace(*prod.ConvertionUnit) != "" {
|
||||
satuan = *prod.ConvertionUnit
|
||||
}
|
||||
|
||||
if err := file.SetCellValue(sheet, "A"+r, soNumber); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "B"+r, soDate); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "C"+r, status); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "D"+r, customer); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "E"+r, safeMarketingExportText(prod.MarketingType)); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "F"+r, productName); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "G"+r, week); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "H"+r, prod.Qty); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "I"+r, satuan); err != nil {
|
||||
return err
|
||||
}
|
||||
if prod.TotalPeti != nil {
|
||||
if err := file.SetCellValue(sheet, "J"+r, *prod.TotalPeti); err != nil {
|
||||
return err
|
||||
}
|
||||
} else {
|
||||
if err := file.SetCellValue(sheet, "J"+r, "-"); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "K"+r, prod.AvgWeight); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "L"+r, prod.TotalWeight); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "M"+r, prod.UnitPrice); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "N"+r, prod.TotalPrice); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "O"+r, grandTotal); err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellValue(sheet, "P"+r, notes); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
lastRow := len(items) + 1
|
||||
lastRow := row
|
||||
lastRowStr := strconv.Itoa(lastRow)
|
||||
border := []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
}
|
||||
|
||||
dataStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{
|
||||
Horizontal: "left",
|
||||
Vertical: "center",
|
||||
WrapText: true,
|
||||
},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
},
|
||||
Alignment: &excelize.Alignment{Horizontal: "left", Vertical: "center", WrapText: true},
|
||||
Border: border,
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
if err := file.SetCellStyle(sheet, "A2", "G"+strconv.Itoa(lastRow), dataStyle); err != nil {
|
||||
if err := file.SetCellStyle(sheet, "A2", "P"+lastRowStr, dataStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
moneyStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{
|
||||
Horizontal: "right",
|
||||
Vertical: "center",
|
||||
},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "D1D5DB", Style: 1},
|
||||
{Type: "top", Color: "D1D5DB", Style: 1},
|
||||
{Type: "bottom", Color: "D1D5DB", Style: 1},
|
||||
{Type: "right", Color: "D1D5DB", Style: 1},
|
||||
},
|
||||
numberStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{Horizontal: "right", Vertical: "center"},
|
||||
Border: border,
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellStyle(sheet, "K2", "O"+lastRowStr, numberStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return file.SetCellStyle(sheet, "E2", "E"+strconv.Itoa(lastRow), moneyStyle)
|
||||
centerStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{Horizontal: "center", Vertical: "center"},
|
||||
Border: border,
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
for _, col := range []string{"G", "H", "J"} {
|
||||
if err := file.SetCellStyle(sheet, col+"2", col+lastRowStr, centerStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func formatMarketingExportDate(value time.Time) string {
|
||||
@@ -226,36 +327,6 @@ func formatMarketingExportStatus(item dto.MarketingListDTO) string {
|
||||
return safeMarketingExportText(item.LatestApproval.StepName)
|
||||
}
|
||||
|
||||
func formatMarketingProducts(items []dto.DeliveryMarketingProductDTO) string {
|
||||
if len(items) == 0 {
|
||||
return "-"
|
||||
}
|
||||
|
||||
seen := make(map[string]struct{})
|
||||
names := make([]string, 0, len(items))
|
||||
for _, item := range items {
|
||||
if item.ProductWarehouse == nil || item.ProductWarehouse.Product == nil {
|
||||
continue
|
||||
}
|
||||
|
||||
name := strings.TrimSpace(item.ProductWarehouse.Product.Name)
|
||||
if name == "" {
|
||||
continue
|
||||
}
|
||||
|
||||
if _, exists := seen[name]; exists {
|
||||
continue
|
||||
}
|
||||
seen[name] = struct{}{}
|
||||
names = append(names, name)
|
||||
}
|
||||
|
||||
if len(names) == 0 {
|
||||
return "-"
|
||||
}
|
||||
|
||||
return strings.Join(names, ", ")
|
||||
}
|
||||
|
||||
func sumMarketingGrandTotal(items []dto.DeliveryMarketingProductDTO) float64 {
|
||||
total := 0.0
|
||||
@@ -266,40 +337,6 @@ func sumMarketingGrandTotal(items []dto.DeliveryMarketingProductDTO) float64 {
|
||||
return total
|
||||
}
|
||||
|
||||
func formatMarketingRupiah(value float64) string {
|
||||
if math.IsNaN(value) || math.IsInf(value, 0) {
|
||||
return "Rp 0"
|
||||
}
|
||||
|
||||
rounded := int64(math.Round(value))
|
||||
sign := ""
|
||||
if rounded < 0 {
|
||||
sign = "-"
|
||||
rounded = -rounded
|
||||
}
|
||||
|
||||
raw := strconv.FormatInt(rounded, 10)
|
||||
if raw == "" {
|
||||
raw = "0"
|
||||
}
|
||||
|
||||
var grouped strings.Builder
|
||||
rem := len(raw) % 3
|
||||
if rem > 0 {
|
||||
grouped.WriteString(raw[:rem])
|
||||
if len(raw) > rem {
|
||||
grouped.WriteString(".")
|
||||
}
|
||||
}
|
||||
for i := rem; i < len(raw); i += 3 {
|
||||
grouped.WriteString(raw[i : i+3])
|
||||
if i+3 < len(raw) {
|
||||
grouped.WriteString(".")
|
||||
}
|
||||
}
|
||||
|
||||
return "Rp " + sign + grouped.String()
|
||||
}
|
||||
|
||||
func safeMarketingExportText(value string) string {
|
||||
trimmed := strings.TrimSpace(value)
|
||||
|
||||
@@ -29,6 +29,7 @@ type MarketingListDTO struct {
|
||||
SalesPerson userDTO.UserRelationDTO `json:"sales_person"`
|
||||
SoDocs string `json:"so_docs"`
|
||||
SalesOrder []DeliveryMarketingProductDTO `json:"sales_order"`
|
||||
DeliveryOrder []DeliveryGroupDTO `json:"delivery_order"`
|
||||
CreatedUser userDTO.UserRelationDTO `json:"created_user"`
|
||||
CreatedAt time.Time `json:"created_at"`
|
||||
UpdatedAt time.Time `json:"updated_at"`
|
||||
@@ -203,6 +204,7 @@ func ToMarketingListDTO(marketing *entity.Marketing, deliveryProducts []entity.M
|
||||
SalesPerson: salesPerson,
|
||||
SoDocs: marketing.SoDocs,
|
||||
SalesOrder: salesOrderProducts,
|
||||
DeliveryOrder: extractDeliveryGroupsFromProducts(marketing),
|
||||
CreatedUser: createdUser,
|
||||
CreatedAt: marketing.CreatedAt,
|
||||
UpdatedAt: marketing.UpdatedAt,
|
||||
@@ -376,6 +378,23 @@ func GenerateDeliveryOrderNumber(soNumber string, deliveryDate *time.Time, wareh
|
||||
return numberPrefix
|
||||
}
|
||||
|
||||
func extractDeliveryGroupsFromProducts(marketing *entity.Marketing) []DeliveryGroupDTO {
|
||||
var dps []MarketingDeliveryProductDTO
|
||||
for _, product := range marketing.Products {
|
||||
if product.DeliveryProduct == nil || product.DeliveryProduct.DeliveryDate == nil {
|
||||
continue
|
||||
}
|
||||
dp := ToMarketingDeliveryProductDTO(*product.DeliveryProduct)
|
||||
if product.ProductWarehouse.Id != 0 {
|
||||
mapped := productwarehouseDTO.ToProductWarehouseNestedDTO(product.ProductWarehouse)
|
||||
dp.ProductWarehouse = &mapped
|
||||
}
|
||||
dp.ConvertionUnit = product.ConvertionUnit
|
||||
dps = append(dps, dp)
|
||||
}
|
||||
return groupDeliveryProducts(dps, marketing.SoNumber)
|
||||
}
|
||||
|
||||
func collectDoNumbers(marketing *entity.Marketing) []string {
|
||||
if marketing == nil || len(marketing.Products) == 0 {
|
||||
return nil
|
||||
|
||||
@@ -292,7 +292,29 @@ func (s deliveryOrdersService) GetAll(c *fiber.Ctx, params *validation.DeliveryO
|
||||
if params.MarketingId != 0 {
|
||||
return db.Where("id = ?", params.MarketingId)
|
||||
}
|
||||
return db.Order("created_at DESC").Order("updated_at DESC")
|
||||
|
||||
orderDir := "DESC"
|
||||
if params.SortOrder != "" {
|
||||
orderDir = strings.ToUpper(params.SortOrder)
|
||||
}
|
||||
|
||||
switch strings.TrimSpace(params.SortBy) {
|
||||
case "so_number":
|
||||
return db.Order("marketings.so_number " + orderDir)
|
||||
case "so_date":
|
||||
return db.Order("marketings.so_date " + orderDir)
|
||||
case "status":
|
||||
statusSQL := "(SELECT step_name FROM approvals WHERE approvable_type = '" + utils.ApprovalWorkflowMarketing.String() + "' AND approvable_id = marketings.id ORDER BY action_at DESC, id DESC LIMIT 1) " + orderDir
|
||||
return db.Order(statusSQL)
|
||||
case "customer":
|
||||
return db.Joins("LEFT JOIN customers ON customers.id = marketings.customer_id").Order("COALESCE(customers.name, '') " + orderDir)
|
||||
case "grand_total":
|
||||
return db.Order("(SELECT COALESCE(SUM(mp.total_price), 0) FROM marketing_products mp WHERE mp.marketing_id = marketings.id) " + orderDir)
|
||||
case "created_at":
|
||||
return db.Order("marketings.created_at " + orderDir)
|
||||
default:
|
||||
return db.Order("created_at DESC").Order("updated_at DESC")
|
||||
}
|
||||
})
|
||||
if err != nil {
|
||||
return nil, 0, err
|
||||
@@ -520,9 +542,15 @@ func (s deliveryOrdersService) UpdateOne(c *fiber.Ctx, req *validation.DeliveryO
|
||||
return nil, err
|
||||
}
|
||||
|
||||
latestApproval, err := s.ApprovalSvc.LatestByTarget(c.Context(), utils.ApprovalWorkflowMarketing, id, nil)
|
||||
if err != nil {
|
||||
return nil, fiber.NewError(fiber.StatusInternalServerError, "Failed to check approval status")
|
||||
}
|
||||
|
||||
err = s.MarketingRepo.DB().WithContext(c.Context()).Transaction(func(dbTransaction *gorm.DB) error {
|
||||
marketingProductRepositoryTx := marketingRepo.NewMarketingProductRepository(dbTransaction)
|
||||
marketingDeliveryProductRepositoryTx := marketingRepo.NewMarketingDeliveryProductRepository(dbTransaction)
|
||||
approvalSvcTx := commonSvc.NewApprovalService(commonRepo.NewApprovalRepository(dbTransaction))
|
||||
marketingRepoTx := marketingRepo.NewMarketingRepository(dbTransaction)
|
||||
|
||||
marketing, err := marketingRepoTx.GetByID(c.Context(), id, nil)
|
||||
@@ -608,6 +636,23 @@ func (s deliveryOrdersService) UpdateOne(c *fiber.Ctx, req *validation.DeliveryO
|
||||
}
|
||||
}
|
||||
|
||||
if latestApproval != nil && latestApproval.StepNumber == uint16(utils.MarketingDeliveryOrder) {
|
||||
action := entity.ApprovalActionUpdated
|
||||
_, err := approvalSvcTx.CreateApproval(
|
||||
c.Context(),
|
||||
utils.ApprovalWorkflowMarketing,
|
||||
id,
|
||||
utils.MarketingStepSalesOrder,
|
||||
&action,
|
||||
actorID,
|
||||
nil)
|
||||
if err != nil {
|
||||
if !errors.Is(err, gorm.ErrDuplicatedKey) {
|
||||
return fiber.NewError(fiber.StatusInternalServerError, "Failed to reset approval to Sales Order")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return nil
|
||||
})
|
||||
if err != nil {
|
||||
|
||||
@@ -516,7 +516,7 @@ func (s salesOrdersService) UpdateOne(c *fiber.Ctx, req *validation.Update, id u
|
||||
c.Context(),
|
||||
utils.ApprovalWorkflowMarketing,
|
||||
id,
|
||||
approvalutils.ApprovalStep(latestApproval.StepNumber),
|
||||
utils.MarketingStepPengajuan,
|
||||
&action,
|
||||
actorID,
|
||||
nil)
|
||||
@@ -770,15 +770,21 @@ func (s salesOrdersService) Approval(c *fiber.Ctx, req *validation.Approve) ([]e
|
||||
|
||||
func (s *salesOrdersService) createMarketingProductWithDelivery(ctx context.Context, marketingId uint, marketingType string, rp validation.CreateMarketingProduct, marketingProductRepo repository.MarketingProductRepository, invDeliveryRepo repository.MarketingDeliveryProductRepository) error {
|
||||
|
||||
totalWeight, totalPrice := s.calculatePriceByMarketingType(
|
||||
marketingType,
|
||||
rp.Qty,
|
||||
rp.AvgWeight,
|
||||
rp.UnitPrice,
|
||||
rp.Week,
|
||||
rp.ConvertionUnit,
|
||||
rp.WeightPerConvertion,
|
||||
)
|
||||
var totalWeight, totalPrice float64
|
||||
if rp.TotalPrice != nil {
|
||||
totalWeight = math.Round(rp.Qty*rp.AvgWeight*100) / 100
|
||||
totalPrice = *rp.TotalPrice
|
||||
} else {
|
||||
totalWeight, totalPrice = s.calculatePriceByMarketingType(
|
||||
marketingType,
|
||||
rp.Qty,
|
||||
rp.AvgWeight,
|
||||
rp.UnitPrice,
|
||||
rp.Week,
|
||||
rp.ConvertionUnit,
|
||||
rp.WeightPerConvertion,
|
||||
)
|
||||
}
|
||||
|
||||
marketingProduct := &entity.MarketingProduct{
|
||||
MarketingId: marketingId,
|
||||
@@ -821,7 +827,7 @@ func (s *salesOrdersService) calculatePriceByMarketingType(marketingType string,
|
||||
totalPrice = math.Round(qty*unitPrice*100) / 100
|
||||
} else if marketingType == string(utils.MarketingTypeAyamPullet) && week != nil && *week > 0 {
|
||||
totalWeight = math.Round(qty*avgWeight*100) / 100
|
||||
totalPrice = math.Round(unitPrice*float64(*week)*qty*100) / 100
|
||||
totalPrice = math.Round(totalWeight*unitPrice*100) / 100
|
||||
} else {
|
||||
totalWeight = math.Round(qty*avgWeight*100) / 100
|
||||
|
||||
|
||||
@@ -31,6 +31,8 @@ type DeliveryOrderQuery struct {
|
||||
MarketingId uint `query:"marketing_id" validate:"omitempty,gt=0"`
|
||||
ProjectFlockID uint `query:"project_flock_id" validate:"omitempty,gt=0"`
|
||||
ProjectFlockKandangID uint `query:"project_flock_kandang_id" validate:"omitempty,gt=0"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,oneof=so_number so_date status customer grand_total created_at"`
|
||||
SortOrder string `query:"sort_order" validate:"omitempty,oneof=asc desc"`
|
||||
}
|
||||
|
||||
type DeliveryOrderApprove struct {
|
||||
|
||||
@@ -26,6 +26,7 @@ type CreateMarketingProduct struct {
|
||||
UnitPrice float64 `json:"unit_price" validate:"required,gt=0"`
|
||||
Qty float64 `json:"qty" validate:"required,gt=0"`
|
||||
AvgWeight float64 `json:"avg_weight" validate:"omitempty,gt=0"`
|
||||
TotalPrice *float64 `json:"total_price" validate:"omitempty,gt=0"`
|
||||
}
|
||||
|
||||
type Update struct {
|
||||
|
||||
@@ -14,6 +14,7 @@ type CustomerRelationDTO struct {
|
||||
Name string `json:"name"`
|
||||
Type string `json:"type"`
|
||||
AccountNumber string `json:"account_number"`
|
||||
BankName string `json:"bank_name"`
|
||||
Address string `json:"address,omitempty"`
|
||||
Balance float64 `json:"balance"`
|
||||
Pic *userDTO.UserRelationDTO `json:"pic,omitempty"`
|
||||
@@ -28,6 +29,7 @@ type CustomerListDTO struct {
|
||||
Phone string `json:"phone"`
|
||||
Email string `json:"email"`
|
||||
AccountNumber string `json:"account_number"`
|
||||
BankName string `json:"bank_name"`
|
||||
Balance float64 `json:"balance"`
|
||||
Pic userDTO.UserRelationDTO `json:"pic"`
|
||||
CreatedUser userDTO.UserRelationDTO `json:"created_user"`
|
||||
@@ -53,6 +55,7 @@ func ToCustomerRelationDTO(e entity.Customer) CustomerRelationDTO {
|
||||
Name: e.Name,
|
||||
Type: e.Type,
|
||||
AccountNumber: e.AccountNumber,
|
||||
BankName: e.BankName,
|
||||
Address: e.Address,
|
||||
Balance: e.Balance,
|
||||
Pic: pic,
|
||||
@@ -81,6 +84,7 @@ func ToCustomerListDTO(e entity.Customer) CustomerListDTO {
|
||||
Phone: e.Phone,
|
||||
Email: e.Email,
|
||||
AccountNumber: e.AccountNumber,
|
||||
BankName: e.BankName,
|
||||
Pic: pic,
|
||||
CreatedAt: e.CreatedAt,
|
||||
UpdatedAt: e.UpdatedAt,
|
||||
|
||||
@@ -133,6 +133,7 @@ func (s *customerService) CreateOne(c *fiber.Ctx, req *validation.Create) (*enti
|
||||
Phone: req.Phone,
|
||||
Email: req.Email,
|
||||
AccountNumber: req.AccountNumber,
|
||||
BankName: req.BankName,
|
||||
CreatedBy: actorID,
|
||||
}
|
||||
|
||||
@@ -193,6 +194,10 @@ func (s customerService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint
|
||||
updateBody["account_number"] = *req.AccountNumber
|
||||
}
|
||||
|
||||
if req.BankName != nil {
|
||||
updateBody["bank_name"] = *req.BankName
|
||||
}
|
||||
|
||||
if len(updateBody) == 0 {
|
||||
return s.GetOne(c, id)
|
||||
}
|
||||
|
||||
@@ -8,6 +8,7 @@ type Create struct {
|
||||
Phone string `json:"phone" validate:"required_strict,max=20"`
|
||||
Email string `json:"email" validate:"required_strict,email,max=50"`
|
||||
AccountNumber string `json:"account_number" validate:"required_strict,max=50"`
|
||||
BankName string `json:"bank_name" validate:"required_strict,max=100"`
|
||||
}
|
||||
|
||||
type Update struct {
|
||||
@@ -18,6 +19,7 @@ type Update struct {
|
||||
Phone *string `json:"phone,omitempty" validate:"omitempty,max=20"`
|
||||
Email *string `json:"email,omitempty" validate:"omitempty,max=50"`
|
||||
AccountNumber *string `json:"account_number,omitempty" validate:"omitempty,max=50"`
|
||||
BankName *string `json:"bank_name,omitempty" validate:"omitempty,max=100"`
|
||||
}
|
||||
|
||||
type Query struct {
|
||||
|
||||
@@ -24,9 +24,11 @@ func NewEmployeesController(employeesService service.EmployeesService) *Employee
|
||||
|
||||
func (u *EmployeesController) GetAll(c *fiber.Ctx) error {
|
||||
query := &validation.Query{
|
||||
Page: c.QueryInt("page", 1),
|
||||
Limit: c.QueryInt("limit", 10),
|
||||
Search: c.Query("search", ""),
|
||||
Page: c.QueryInt("page", 1),
|
||||
Limit: c.QueryInt("limit", 10),
|
||||
Search: c.Query("search", ""),
|
||||
OrderBy: c.Query("order_by", "desc"),
|
||||
SortBy: c.Query("sort_by", "updated_at"),
|
||||
}
|
||||
|
||||
if query.Page < 1 || query.Limit < 1 {
|
||||
|
||||
@@ -2,6 +2,7 @@ package service
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
|
||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||
@@ -126,11 +127,18 @@ func (s employeesService) GetAll(c *fiber.Ctx, params *validation.Query) ([]enti
|
||||
if params.IsActive != nil {
|
||||
db = db.Where("employees.is_active = ?", *params.IsActive)
|
||||
}
|
||||
return db.
|
||||
|
||||
db = db.
|
||||
Select("employees.id, employees.name, employees.is_active, employees.created_at, employees.updated_at").
|
||||
Group("employees.id, employees.name, employees.is_active, employees.created_at, employees.updated_at").
|
||||
Order("employees.created_at DESC").
|
||||
Order("employees.updated_at DESC")
|
||||
Group("employees.id, employees.name, employees.is_active, employees.created_at, employees.updated_at")
|
||||
|
||||
if params.OrderBy == "desc" || params.OrderBy == "" {
|
||||
db = db.Order(fmt.Sprintf("employees.%s DESC", params.SortBy))
|
||||
} else {
|
||||
db = db.Order(fmt.Sprintf("employees.%s ASC", params.SortBy))
|
||||
}
|
||||
|
||||
return db
|
||||
})
|
||||
|
||||
if err != nil {
|
||||
|
||||
@@ -18,4 +18,6 @@ type Query struct {
|
||||
Search string `query:"search" validate:"omitempty,max=50"`
|
||||
KandangId *uint `query:"kandang_id" validate:"omitempty"`
|
||||
IsActive *bool `query:"is_active" validate:"omitempty"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,max=50,oneof=name created_at updated_at" default:"updated_at"`
|
||||
OrderBy string `query:"order_by" validate:"omitempty,oneof=asc desc" default:"desc"`
|
||||
}
|
||||
|
||||
@@ -29,6 +29,8 @@ func (u *KandangGroupController) GetAll(c *fiber.Ctx) error {
|
||||
Search: c.Query("search", ""),
|
||||
LocationId: c.QueryInt("location_id", 0),
|
||||
PicId: c.QueryInt("pic_id", 0),
|
||||
OrderBy: c.Query("order_by", "desc"),
|
||||
SortBy: c.Query("sort_by", "updated_at"),
|
||||
}
|
||||
|
||||
if query.Page < 1 || query.Limit < 1 {
|
||||
|
||||
@@ -70,7 +70,14 @@ func (s kandangGroupService) GetAll(c *fiber.Ctx, params *validation.Query) ([]e
|
||||
if params.PicId != 0 {
|
||||
db = db.Where("kandang_groups.pic_id = ?", params.PicId)
|
||||
}
|
||||
return db.Order("kandang_groups.created_at DESC").Order("kandang_groups.updated_at DESC")
|
||||
|
||||
if params.OrderBy == "desc" || params.OrderBy == "" {
|
||||
db = db.Order(fmt.Sprintf("kandang_groups.%s DESC", params.SortBy))
|
||||
} else {
|
||||
db = db.Order(fmt.Sprintf("kandang_groups.%s ASC", params.SortBy))
|
||||
}
|
||||
|
||||
return db
|
||||
})
|
||||
|
||||
if scopeErr != nil {
|
||||
|
||||
@@ -20,4 +20,6 @@ type Query struct {
|
||||
Search string `query:"search" validate:"omitempty,max=50"`
|
||||
LocationId int `query:"location_id" validate:"omitempty,number,gt=0"`
|
||||
PicId int `query:"pic_id" validate:"omitempty,number,gt=0"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,max=50,oneof=name created_at updated_at" default:"updated_at"`
|
||||
OrderBy string `query:"order_by" validate:"omitempty,oneof=asc desc" default:"desc"`
|
||||
}
|
||||
|
||||
@@ -29,6 +29,8 @@ func (u *KandangController) GetAll(c *fiber.Ctx) error {
|
||||
Search: c.Query("search", ""),
|
||||
LocationId: c.QueryInt("location_id", 0),
|
||||
PicId: c.QueryInt("pic_id", 0),
|
||||
OrderBy: c.Query("order_by", "desc"),
|
||||
SortBy: c.Query("sort_by", "created_at"),
|
||||
}
|
||||
|
||||
if query.Page < 1 || query.Limit < 1 {
|
||||
|
||||
@@ -66,7 +66,14 @@ func (s kandangService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity
|
||||
if params.PicId != 0 {
|
||||
db = db.Where("pic_id = ?", params.PicId)
|
||||
}
|
||||
return db.Order("created_at DESC").Order("updated_at DESC")
|
||||
|
||||
if params.OrderBy == "desc" || params.OrderBy == "" {
|
||||
db = db.Order(fmt.Sprintf("%s DESC", params.SortBy))
|
||||
} else {
|
||||
db = db.Order(fmt.Sprintf("%s ASC", params.SortBy))
|
||||
}
|
||||
|
||||
return db
|
||||
})
|
||||
|
||||
if scopeErr != nil {
|
||||
|
||||
@@ -26,4 +26,6 @@ type Query struct {
|
||||
Search string `query:"search" validate:"omitempty,max=50"`
|
||||
LocationId int `query:"location_id" validate:"omitempty,number,gt=0"`
|
||||
PicId int `query:"pic_id" validate:"omitempty,number,gt=0"`
|
||||
SortBy string `query:"sort_by" validate:"omitempty,max=50,oneof=name created_at updated_at" default:"updated_at"`
|
||||
OrderBy string `query:"order_by" validate:"omitempty,oneof=asc desc" default:"desc"`
|
||||
}
|
||||
|
||||
@@ -26,6 +26,7 @@ type SupplierListDTO struct {
|
||||
Address string `json:"address"`
|
||||
Npwp *string `json:"npwp,omitempty"`
|
||||
AccountNumber *string `json:"account_number,omitempty"`
|
||||
BankName *string `json:"bank_name,omitempty"`
|
||||
Balance float64 `json:"balance"`
|
||||
DueDate int `json:"due_date"`
|
||||
CreatedUser *userDTO.UserRelationDTO `json:"created_user"`
|
||||
@@ -66,6 +67,7 @@ func ToSupplierListDTO(e entity.Supplier) SupplierListDTO {
|
||||
Address: e.Address,
|
||||
Npwp: e.Npwp,
|
||||
AccountNumber: e.AccountNumber,
|
||||
BankName: e.BankName,
|
||||
Balance: e.Balance,
|
||||
DueDate: e.DueDate,
|
||||
SupplierRelationDTO: ToSupplierRelationDTO(e),
|
||||
|
||||
@@ -160,6 +160,7 @@ func (s *supplierService) CreateOne(c *fiber.Ctx, req *validation.Create) (*enti
|
||||
Address: req.Address,
|
||||
Npwp: req.Npwp,
|
||||
AccountNumber: req.AccountNumber,
|
||||
BankName: req.BankName,
|
||||
DueDate: req.DueDate,
|
||||
CreatedBy: actorID,
|
||||
}
|
||||
@@ -243,6 +244,10 @@ func (s supplierService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint
|
||||
updateBody["account_number"] = *req.AccountNumber
|
||||
}
|
||||
|
||||
if req.BankName != nil {
|
||||
updateBody["bank_name"] = *req.BankName
|
||||
}
|
||||
|
||||
if req.DueDate != nil {
|
||||
updateBody["due_date"] = *req.DueDate
|
||||
}
|
||||
|
||||
@@ -12,6 +12,7 @@ type Create struct {
|
||||
Address string `json:"address" validate:"required_strict"`
|
||||
Npwp *string `json:"npwp,omitempty" validate:"omitempty,max=50"`
|
||||
AccountNumber *string `json:"account_number,omitempty" validate:"omitempty,max=50"`
|
||||
BankName *string `json:"bank_name,omitempty" validate:"omitempty,max=100"`
|
||||
DueDate int `json:"due_date" validate:"required_strict,number,gt=0"`
|
||||
}
|
||||
|
||||
@@ -27,6 +28,7 @@ type Update struct {
|
||||
Address *string `json:"address,omitempty" validate:"omitempty"`
|
||||
Npwp *string `json:"npwp,omitempty" validate:"omitempty,max=50"`
|
||||
AccountNumber *string `json:"account_number,omitempty" validate:"omitempty,max=50"`
|
||||
BankName *string `json:"bank_name,omitempty" validate:"omitempty,max=100"`
|
||||
DueDate *int `json:"due_date,omitempty" validate:"omitempty,number,gt=0"`
|
||||
}
|
||||
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
package controller
|
||||
|
||||
import (
|
||||
"math"
|
||||
"strconv"
|
||||
|
||||
"gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/dto"
|
||||
@@ -21,32 +22,32 @@ func NewChickinController(chickinService service.ChickinService) *ChickinControl
|
||||
}
|
||||
}
|
||||
|
||||
// func (u *ChickinController) GetAll(c *fiber.Ctx) error {
|
||||
// query := &validation.Query{
|
||||
// Page: c.QueryInt("page", 1),
|
||||
// Limit: c.QueryInt("limit", 10),
|
||||
// ProjectFlockKandangId: uint(c.QueryInt("project_flock_kandang_id", 0)),
|
||||
// }
|
||||
func (u *ChickinController) GetAll(c *fiber.Ctx) error {
|
||||
query := &validation.Query{
|
||||
Page: c.QueryInt("page", 1),
|
||||
Limit: c.QueryInt("limit", 10),
|
||||
ProjectFlockKandangId: uint(c.QueryInt("project_flock_kandang_id", 0)),
|
||||
}
|
||||
|
||||
// result, totalResults, err := u.ChickinService.GetAll(c, query)
|
||||
// if err != nil {
|
||||
// return err
|
||||
// }
|
||||
result, totalResults, err := u.ChickinService.GetAll(c, query)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
// return c.Status(fiber.StatusOK).
|
||||
// JSON(response.SuccessWithPaginate[dto.ChickinListDTO]{
|
||||
// Code: fiber.StatusOK,
|
||||
// Status: "success",
|
||||
// Message: "Get all chickins successfully",
|
||||
// Meta: response.Meta{
|
||||
// Page: query.Page,
|
||||
// Limit: query.Limit,
|
||||
// TotalPages: int64(math.Ceil(float64(totalResults) / float64(query.Limit))),
|
||||
// TotalResults: totalResults,
|
||||
// },
|
||||
// Data: dto.ToChickinListDTOs(result),
|
||||
// })
|
||||
// }
|
||||
return c.Status(fiber.StatusOK).
|
||||
JSON(response.SuccessWithPaginate[dto.ChickinListDTO]{
|
||||
Code: fiber.StatusOK,
|
||||
Status: "success",
|
||||
Message: "Get all chickins successfully",
|
||||
Meta: response.Meta{
|
||||
Page: query.Page,
|
||||
Limit: query.Limit,
|
||||
TotalPages: int64(math.Ceil(float64(totalResults) / float64(query.Limit))),
|
||||
TotalResults: totalResults,
|
||||
},
|
||||
Data: dto.ToChickinListDTOs(result),
|
||||
})
|
||||
}
|
||||
|
||||
// func (u *ChickinController) GetOne(c *fiber.Ctx) error {
|
||||
// param := c.Params("id")
|
||||
|
||||
@@ -15,8 +15,8 @@ func ChickinRoutes(v1 fiber.Router, u user.UserService, s chickin.ChickinService
|
||||
route := v1.Group("/chickins")
|
||||
route.Use(m.Auth(u))
|
||||
|
||||
// route.Get("/", ctrl.GetAll)
|
||||
route.Post("/",m.RequirePermissions(m.P_ChickinsCreateOne), ctrl.CreateOne)
|
||||
route.Get("/", m.RequirePermissions(m.P_ChickinsGetAll), ctrl.GetAll)
|
||||
route.Post("/", m.RequirePermissions(m.P_ChickinsCreateOne), ctrl.CreateOne)
|
||||
route.Get("/:id",m.RequirePermissions(m.P_ChickinsGetOne), ctrl.GetOne)
|
||||
// route.Patch("/:id", ctrl.UpdateOne)
|
||||
route.Delete("/:id", ctrl.DeleteOne)
|
||||
|
||||
@@ -312,10 +312,10 @@ func (u *ProjectflockController) LookupProjectFlockKandang(c *fiber.Ctx) error {
|
||||
mapped := warehouseDTO.ToWarehouseRelationDTO(*warehouse)
|
||||
dtoResult.Warehouse = &mapped
|
||||
}
|
||||
if _, isLaying, serr := u.ProjectflockService.GetProjectFlockKandangTransferStateAtDate(c, result.Id, recordDate); serr != nil {
|
||||
if isTransition, isLaying, serr := u.ProjectflockService.GetProjectFlockKandangTransferStateAtDate(c, result.Id, recordDate); serr != nil {
|
||||
return serr
|
||||
} else {
|
||||
dtoResult.IsTransition = false
|
||||
dtoResult.IsTransition = isTransition
|
||||
dtoResult.IsLaying = isLaying
|
||||
}
|
||||
applyCutOverLayingLookupOverride(&dtoResult)
|
||||
@@ -346,7 +346,7 @@ func (u *ProjectflockController) LookupProjectFlockKandang(c *fiber.Ctx) error {
|
||||
}
|
||||
|
||||
func applyCutOverLayingLookupOverride(result *dto.ProjectFlockKandangDTO) {
|
||||
if result == nil || result.ProjectFlock == nil || result.IsLaying || result.ChickInDate == nil {
|
||||
if result == nil || result.ProjectFlock == nil || result.IsLaying || result.IsTransition || result.ChickInDate == nil {
|
||||
return
|
||||
}
|
||||
|
||||
|
||||
@@ -588,17 +588,29 @@ func (s projectflockService) GetProjectFlockKandangTransferStateAtDate(ctx *fibe
|
||||
switch category {
|
||||
case strings.ToUpper(string(utils.ProjectFlockCategoryGrowing)):
|
||||
transfer, err = s.TransferLayingRepo.GetLatestApprovedBySourceKandang(ctx.Context(), projectFlockKandangID)
|
||||
if err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
return false, false, nil
|
||||
}
|
||||
s.Log.Errorf("Failed to resolve transfer state for project flock kandang %d: %+v", projectFlockKandangID, err)
|
||||
return false, false, fiber.NewError(fiber.StatusInternalServerError, "Failed to resolve transfer state")
|
||||
}
|
||||
case strings.ToUpper(string(utils.ProjectFlockCategoryLaying)):
|
||||
transfer, err = s.TransferLayingRepo.GetLatestApprovedByTargetKandang(ctx.Context(), projectFlockKandangID)
|
||||
default:
|
||||
return false, false, nil
|
||||
}
|
||||
if err != nil {
|
||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||
// Multi-source: target kandang bisa menerima dari multiple transfer terpisah. Pakai
|
||||
// EARLIEST transfer (transfer_date ASC) sebagai anchor — kandang masuk transition/laying
|
||||
// mengikuti batch pertama yang sampai.
|
||||
allTransfers, allErr := s.TransferLayingRepo.GetAllApprovedByTargetKandang(ctx.Context(), projectFlockKandangID)
|
||||
if allErr != nil {
|
||||
s.Log.Errorf("Failed to resolve transfers for project flock kandang %d: %+v", projectFlockKandangID, allErr)
|
||||
return false, false, fiber.NewError(fiber.StatusInternalServerError, "Failed to resolve transfer state")
|
||||
}
|
||||
if len(allTransfers) == 0 {
|
||||
return false, false, nil
|
||||
}
|
||||
s.Log.Errorf("Failed to resolve transfer state for project flock kandang %d: %+v", projectFlockKandangID, err)
|
||||
return false, false, fiber.NewError(fiber.StatusInternalServerError, "Failed to resolve transfer state")
|
||||
// Repository ORDER BY transfer_date ASC, id ASC → [0] = earliest
|
||||
transfer = &allTransfers[0]
|
||||
default:
|
||||
return false, false, nil
|
||||
}
|
||||
if transfer == nil {
|
||||
return false, false, nil
|
||||
|
||||
@@ -77,6 +77,8 @@ func setRecordingExportColumns(file *excelize.File, sheet string) error {
|
||||
"Z": 22,
|
||||
"AA": 16,
|
||||
"AB": 18,
|
||||
"AC": 24,
|
||||
"AD": 18,
|
||||
}
|
||||
|
||||
for col, width := range columnWidths {
|
||||
@@ -96,7 +98,7 @@ func setRecordingExportColumns(file *excelize.File, sheet string) error {
|
||||
}
|
||||
|
||||
func setRecordingExportHeaders(file *excelize.File, sheet string) error {
|
||||
verticalHeaderCols := []string{"A", "B", "C", "D", "E", "F", "G", "H", "I", "Y", "Z", "AA", "AB"}
|
||||
verticalHeaderCols := []string{"A", "B", "C", "D", "E", "F", "G", "H", "I", "Y", "Z", "AA", "AB", "AC", "AD"}
|
||||
for _, col := range verticalHeaderCols {
|
||||
if err := file.MergeCell(sheet, col+"1", col+"2"); err != nil {
|
||||
return err
|
||||
@@ -104,19 +106,21 @@ func setRecordingExportHeaders(file *excelize.File, sheet string) error {
|
||||
}
|
||||
|
||||
headerValues := map[string]string{
|
||||
"A1": "No",
|
||||
"B1": "Lokasi",
|
||||
"C1": "Flock",
|
||||
"D1": "Kandang",
|
||||
"E1": "Periode",
|
||||
"F1": "Kategori",
|
||||
"G1": "Umur (hari)",
|
||||
"H1": "Waktu Recording",
|
||||
"I1": "Populasi Akhir",
|
||||
"Y1": "Status Approval",
|
||||
"Z1": "Catatan Approval",
|
||||
"AA1": "Dibuat Oleh",
|
||||
"AB1": "Tanggal Submit",
|
||||
"A1": "No",
|
||||
"B1": "Lokasi",
|
||||
"C1": "Flock",
|
||||
"D1": "Kandang",
|
||||
"E1": "Periode",
|
||||
"F1": "Kategori",
|
||||
"G1": "Umur (hari)",
|
||||
"H1": "Waktu Recording",
|
||||
"I1": "Populasi Akhir",
|
||||
"Y1": "Status Approval",
|
||||
"Z1": "Catatan Approval",
|
||||
"AA1": "Dibuat Oleh",
|
||||
"AB1": "Tanggal Submit",
|
||||
"AC1": "Nama Sapronak",
|
||||
"AD1": "Jumlah Input Sapronak",
|
||||
}
|
||||
for cell, value := range headerValues {
|
||||
if err := file.SetCellValue(sheet, cell, value); err != nil {
|
||||
@@ -230,7 +234,7 @@ func setRecordingExportHeaders(file *excelize.File, sheet string) error {
|
||||
return err
|
||||
}
|
||||
|
||||
return file.SetCellStyle(sheet, "A1", "AB2", headerStyle)
|
||||
return file.SetCellStyle(sheet, "A1", "AD2", headerStyle)
|
||||
}
|
||||
|
||||
func setRecordingExportRows(file *excelize.File, sheet string, items []dto.RecordingListDTO) error {
|
||||
@@ -241,11 +245,14 @@ func setRecordingExportRows(file *excelize.File, sheet string, items []dto.Recor
|
||||
columns := []string{
|
||||
"A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N",
|
||||
"O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "AA", "AB",
|
||||
"AC", "AD",
|
||||
}
|
||||
|
||||
for i, item := range items {
|
||||
rowNumber := i + 3
|
||||
currentRow := 3
|
||||
type rowRange struct{ start, end int }
|
||||
itemRanges := make([]rowRange, 0, len(items))
|
||||
|
||||
for i, item := range items {
|
||||
fcrStd := 0.0
|
||||
if item.ProjectFlock.Fcr != nil {
|
||||
fcrStd = item.ProjectFlock.Fcr.FcrStd
|
||||
@@ -283,46 +290,79 @@ func setRecordingExportRows(file *excelize.File, sheet string, items []dto.Recor
|
||||
createdBy = safeExportText(item.Approval.ActionBy.Name)
|
||||
}
|
||||
|
||||
rowValues := []interface{}{
|
||||
i + 1,
|
||||
locationName,
|
||||
safeExportText(item.ProjectFlock.FlockName),
|
||||
kandangName,
|
||||
item.ProjectFlock.Period,
|
||||
formatCategoryLabel(item.ProjectFlock.ProjectFlockCategory),
|
||||
formatAgeLabel(item),
|
||||
formatDateIndonesian(item.RecordDatetime),
|
||||
formatNumberID(item.ProjectFlock.TotalChickQty, 0, false),
|
||||
formatNumberID(item.FcrValue, 2, true),
|
||||
formatNumberID(fcrStd, 2, true),
|
||||
formatNumberID(item.FeedIntake, 2, true),
|
||||
formatNumberID(feedIntakeStd, 2, true),
|
||||
formatPercentID(item.CumDepletionRate, 2),
|
||||
formatPercentID(maxDepletionStd, 2),
|
||||
formatNumberID(item.TotalDepletionQty, 2, true),
|
||||
formatNumberID(item.EggMass, 2, true),
|
||||
formatNumberID(eggMassStd, 2, true),
|
||||
formatNumberID(item.EggWeight, 2, true),
|
||||
formatNumberID(eggWeightStd, 2, true),
|
||||
formatPercentID(item.HenDay, 2),
|
||||
formatPercentID(henDayStd, 2),
|
||||
formatPercentID(item.HenHouse, 2),
|
||||
formatPercentID(henHouseStd, 2),
|
||||
formatApprovalStatus(item),
|
||||
safeExportText(pointerString(item.Approval.Notes)),
|
||||
createdBy,
|
||||
formatDateIndonesian(item.CreatedAt),
|
||||
// Expand recordings into one row per sapronak
|
||||
type sapronakRow struct {
|
||||
name string
|
||||
input string
|
||||
}
|
||||
sapronaks := make([]sapronakRow, 0)
|
||||
if len(item.FeedUsage) > 0 {
|
||||
for _, fu := range item.FeedUsage {
|
||||
sapronaks = append(sapronaks, sapronakRow{
|
||||
name: safeExportText(fu.ProductName),
|
||||
input: formatNumberID(fu.UsageAmount+fu.PendingQty, 2, true),
|
||||
})
|
||||
}
|
||||
} else {
|
||||
sapronaks = append(sapronaks, sapronakRow{name: "-", input: "-"})
|
||||
}
|
||||
|
||||
for idx, col := range columns {
|
||||
cell := fmt.Sprintf("%s%d", col, rowNumber)
|
||||
if err := file.SetCellValue(sheet, cell, rowValues[idx]); err != nil {
|
||||
return err
|
||||
groupStart := currentRow
|
||||
|
||||
for sIdx, s := range sapronaks {
|
||||
if sIdx == 0 {
|
||||
rowValues := []interface{}{
|
||||
i + 1, // A
|
||||
locationName, // B
|
||||
safeExportText(item.ProjectFlock.FlockName), // C
|
||||
kandangName, // D
|
||||
item.ProjectFlock.Period, // E
|
||||
formatCategoryLabel(item.ProjectFlock.ProjectFlockCategory), // F
|
||||
formatAgeLabel(item), // G
|
||||
formatDateIndonesian(item.RecordDatetime), // H
|
||||
formatNumberID(item.ProjectFlock.TotalChickQty, 0, false), // I
|
||||
formatNumberID(item.FcrValue, 2, true), // J
|
||||
formatNumberID(fcrStd, 2, true), // K
|
||||
formatNumberID(item.FeedIntake, 2, true), // L
|
||||
formatNumberID(feedIntakeStd, 2, true), // M
|
||||
formatPercentID(item.CumDepletionRate, 2), // N
|
||||
formatPercentID(maxDepletionStd, 2), // O
|
||||
formatNumberID(item.TotalDepletionQty, 2, true), // P
|
||||
formatNumberID(item.EggMass, 2, true), // Q
|
||||
formatNumberID(eggMassStd, 2, true), // R
|
||||
formatNumberID(item.EggWeight, 2, true), // S
|
||||
formatNumberID(eggWeightStd, 2, true), // T
|
||||
formatPercentID(item.HenDay, 2), // U
|
||||
formatPercentID(henDayStd, 2), // V
|
||||
formatPercentID(item.HenHouse, 2), // W
|
||||
formatPercentID(henHouseStd, 2), // X
|
||||
formatApprovalStatus(item), // Y
|
||||
safeExportText(pointerString(item.Approval.Notes)), // Z
|
||||
createdBy, // AA
|
||||
formatDateIndonesian(item.CreatedAt), // AB
|
||||
s.name, // AC
|
||||
s.input, // AD
|
||||
}
|
||||
|
||||
for idx, col := range columns {
|
||||
cell := fmt.Sprintf("%s%d", col, currentRow)
|
||||
if err := file.SetCellValue(sheet, cell, rowValues[idx]); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
} else {
|
||||
file.SetCellValue(sheet, fmt.Sprintf("AC%d", currentRow), s.name)
|
||||
file.SetCellValue(sheet, fmt.Sprintf("AD%d", currentRow), s.input)
|
||||
}
|
||||
|
||||
currentRow++
|
||||
}
|
||||
|
||||
itemRanges = append(itemRanges, rowRange{groupStart, currentRow - 1})
|
||||
}
|
||||
|
||||
lastRow := len(items) + 2
|
||||
lastRow := currentRow - 1
|
||||
|
||||
dataCenterStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{
|
||||
Horizontal: "center",
|
||||
@@ -339,7 +379,7 @@ func setRecordingExportRows(file *excelize.File, sheet string, items []dto.Recor
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if err := file.SetCellStyle(sheet, "A3", fmt.Sprintf("AB%d", lastRow), dataCenterStyle); err != nil {
|
||||
if err := file.SetCellStyle(sheet, "A3", fmt.Sprintf("AD%d", lastRow), dataCenterStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
@@ -360,13 +400,62 @@ func setRecordingExportRows(file *excelize.File, sheet string, items []dto.Recor
|
||||
return err
|
||||
}
|
||||
|
||||
leftColumns := []string{"B", "C", "D", "F", "G", "H", "Y", "Z", "AA", "AB"}
|
||||
leftColumns := []string{"B", "C", "D", "F", "G", "H", "Y", "Z", "AA", "AB", "AC"}
|
||||
for _, col := range leftColumns {
|
||||
if err := file.SetCellStyle(sheet, col+"3", fmt.Sprintf("%s%d", col, lastRow), dataLeftStyle); err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
|
||||
// Apply bottom border on the last sapronak row of each recording group
|
||||
// Separate styles to preserve alignment (AC=left, AD=center) and thin borders
|
||||
borderBottomLeftStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{
|
||||
Horizontal: "left",
|
||||
Vertical: "center",
|
||||
WrapText: true,
|
||||
},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "E6E6E6", Style: 1},
|
||||
{Type: "top", Color: "E6E6E6", Style: 1},
|
||||
{Type: "bottom", Color: "999999", Style: 2},
|
||||
{Type: "right", Color: "E6E6E6", Style: 1},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
borderBottomCenterStyle, err := file.NewStyle(&excelize.Style{
|
||||
Alignment: &excelize.Alignment{
|
||||
Horizontal: "center",
|
||||
Vertical: "center",
|
||||
WrapText: true,
|
||||
},
|
||||
Border: []excelize.Border{
|
||||
{Type: "left", Color: "E6E6E6", Style: 1},
|
||||
{Type: "top", Color: "E6E6E6", Style: 1},
|
||||
{Type: "bottom", Color: "999999", Style: 2},
|
||||
{Type: "right", Color: "E6E6E6", Style: 1},
|
||||
},
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
mergeCols := []string{"A", "B", "C", "D", "E", "F", "G", "H", "I", "J", "K", "L", "M", "N",
|
||||
"O", "P", "Q", "R", "S", "T", "U", "V", "W", "X", "Y", "Z", "AA", "AB",
|
||||
}
|
||||
for _, rng := range itemRanges {
|
||||
if rng.end > rng.start {
|
||||
for _, col := range mergeCols {
|
||||
file.MergeCell(sheet, fmt.Sprintf("%s%d", col, rng.start), fmt.Sprintf("%s%d", col, rng.end))
|
||||
}
|
||||
}
|
||||
file.SetCellStyle(sheet, fmt.Sprintf("AC%d", rng.end), fmt.Sprintf("AC%d", rng.end), borderBottomLeftStyle)
|
||||
file.SetCellStyle(sheet, fmt.Sprintf("AD%d", rng.end), fmt.Sprintf("AD%d", rng.end), borderBottomCenterStyle)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user