mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-05-20 13:31:56 +00:00
Merge branch 'development' into 'staging'
Development See merge request mbugroup/lti-api!192
This commit is contained in:
@@ -0,0 +1,3 @@
|
|||||||
|
ALTER TABLE recording_eggs
|
||||||
|
DROP COLUMN IF EXISTS total_used,
|
||||||
|
DROP COLUMN IF EXISTS total_qty;
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
ALTER TABLE recording_eggs
|
||||||
|
ADD COLUMN total_qty NUMERIC(15, 3) DEFAULT 0 NOT NULL,
|
||||||
|
ADD COLUMN total_used NUMERIC(15, 3) DEFAULT 0 NOT NULL;
|
||||||
|
|
||||||
|
UPDATE recording_eggs
|
||||||
|
SET total_qty = qty
|
||||||
|
WHERE total_qty = 0;
|
||||||
+3
@@ -0,0 +1,3 @@
|
|||||||
|
-- Rollback: add price back to nonstock_suppliers
|
||||||
|
ALTER TABLE nonstock_suppliers
|
||||||
|
ADD COLUMN IF NOT EXISTS price NUMERIC(15, 3) NOT NULL DEFAULT 0;
|
||||||
+3
@@ -0,0 +1,3 @@
|
|||||||
|
-- Migration: remove price from nonstock_suppliers
|
||||||
|
ALTER TABLE nonstock_suppliers
|
||||||
|
DROP COLUMN IF EXISTS price;
|
||||||
@@ -5,7 +5,6 @@ import "time"
|
|||||||
type NonstockSupplier struct {
|
type NonstockSupplier struct {
|
||||||
NonstockId uint `gorm:"not null"`
|
NonstockId uint `gorm:"not null"`
|
||||||
SupplierId uint `gorm:"not null"`
|
SupplierId uint `gorm:"not null"`
|
||||||
Price float64 `gorm:"type:numeric(15,3);not null;default:0"`
|
|
||||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||||
|
|
||||||
Nonstock Nonstock `gorm:"foreignKey:NonstockId;references:Id"`
|
Nonstock Nonstock `gorm:"foreignKey:NonstockId;references:Id"`
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ type RecordingEgg struct {
|
|||||||
RecordingId uint `gorm:"column:recording_id;not null;index"`
|
RecordingId uint `gorm:"column:recording_id;not null;index"`
|
||||||
ProductWarehouseId uint `gorm:"column:product_warehouse_id;not null"`
|
ProductWarehouseId uint `gorm:"column:product_warehouse_id;not null"`
|
||||||
Qty int `gorm:"column:qty;not null"`
|
Qty int `gorm:"column:qty;not null"`
|
||||||
|
TotalQty float64 `gorm:"column:total_qty"`
|
||||||
|
TotalUsed float64 `gorm:"column:total_used"`
|
||||||
Weight *float64 `gorm:"column:weight"`
|
Weight *float64 `gorm:"column:weight"`
|
||||||
CreatedBy uint `gorm:"column:created_by"`
|
CreatedBy uint `gorm:"column:created_by"`
|
||||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
package middleware
|
package middleware
|
||||||
|
|
||||||
const(
|
const (
|
||||||
P_DashboardGetAll = "lti.dashboard.list"
|
P_DashboardGetAll = "lti.dashboard.list"
|
||||||
)
|
)
|
||||||
|
|
||||||
// project-flock
|
// project-flock
|
||||||
const (
|
const (
|
||||||
P_ProjectFlockKandangsClosing = "lti.production.project_flock_kandangs.closing"
|
P_ProjectFlockKandangsClosing = "lti.production.project_flock_kandangs.closing"
|
||||||
@@ -151,7 +152,7 @@ const (
|
|||||||
P_ProductsCreateOne = "lti.master.products.create"
|
P_ProductsCreateOne = "lti.master.products.create"
|
||||||
P_ProductsUpdateOne = "lti.master.products.update"
|
P_ProductsUpdateOne = "lti.master.products.update"
|
||||||
P_ProductsDeleteOne = "lti.master.products.delete"
|
P_ProductsDeleteOne = "lti.master.products.delete"
|
||||||
|
|
||||||
P_SuppliersGetAll = "lti.master.suppliers.list"
|
P_SuppliersGetAll = "lti.master.suppliers.list"
|
||||||
P_SuppliersGetOne = "lti.master.suppliers.detail"
|
P_SuppliersGetOne = "lti.master.suppliers.detail"
|
||||||
P_SuppliersCreateOne = "lti.master.suppliers.create"
|
P_SuppliersCreateOne = "lti.master.suppliers.create"
|
||||||
@@ -238,3 +239,15 @@ const (
|
|||||||
P_UserGetAll = "lti.users.list"
|
P_UserGetAll = "lti.users.list"
|
||||||
P_UserGetOne = "lti.users.detail"
|
P_UserGetOne = "lti.users.detail"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// daily-checklist
|
||||||
|
const (
|
||||||
|
P_DailyChecklistDashboardList = "lti.daily_checklist.dashboard.list"
|
||||||
|
P_DailyChecklistCreateOne = "lti.daily_checklist.create"
|
||||||
|
P_DailyChecklistGetAll = "lti.daily_checklist.list"
|
||||||
|
P_DailyChecklistGetOne = "lti.daily_checklist.detail"
|
||||||
|
P_DailyChecklistReports = "lti.daily_checklist.reports"
|
||||||
|
P_DailyChecklistEmployee = "lti.daily_checklist.master_data.employee"
|
||||||
|
P_DailyChecklistActivity = "lti.daily_checklist.master_data.activity"
|
||||||
|
P_DailyChecklistActivityConfig = "lti.daily_checklist.master_data.configuration"
|
||||||
|
)
|
||||||
|
|||||||
@@ -15,49 +15,49 @@ func DailyChecklistRoutes(v1 fiber.Router, u user.UserService, s dailyChecklist.
|
|||||||
route := v1.Group("/daily-checklists")
|
route := v1.Group("/daily-checklists")
|
||||||
route.Use(m.Auth(u))
|
route.Use(m.Auth(u))
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
route.Get("/", m.RequirePermissions(m.P_DailyChecklistGetAll), ctrl.GetAll)
|
||||||
route.Get("/report", ctrl.GetReport)
|
route.Get("/report", m.RequirePermissions(m.P_DailyChecklistReports), ctrl.GetReport)
|
||||||
|
|
||||||
route.Get("/summary", ctrl.GetSummary)
|
route.Get("/summary", m.RequirePermissions(m.P_DailyChecklistDashboardList), ctrl.GetSummary)
|
||||||
|
|
||||||
route.Get("/report", ctrl.GetReport)
|
// route.Get("/report", ctrl.GetReport)
|
||||||
|
|
||||||
// upsert daily checklist
|
// upsert daily checklist
|
||||||
route.Post("/", ctrl.CreateOne)
|
route.Post("/", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.CreateOne)
|
||||||
|
|
||||||
// get detail data daily checklist by id
|
// get detail data daily checklist by id
|
||||||
route.Get("/relation/:idDailyChecklist", ctrl.GetOne)
|
route.Get("/relation/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistGetOne), ctrl.GetOne)
|
||||||
|
|
||||||
// get phases by daily checklist id
|
// get phases by daily checklist id
|
||||||
route.Get("/phase/:idDailyChecklist", ctrl.GetPhaseByIdChecklist)
|
route.Get("/phase/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.GetPhaseByIdChecklist)
|
||||||
|
|
||||||
// create task
|
// create task
|
||||||
/*
|
/*
|
||||||
ketika add phase
|
ketika add phase
|
||||||
*/
|
*/
|
||||||
route.Post("/phase/:idDailyChecklist", ctrl.CreateDailyChecklistPhase)
|
route.Post("/phase/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.CreateDailyChecklistPhase)
|
||||||
|
|
||||||
// create assigment
|
// create assigment
|
||||||
/*
|
/*
|
||||||
ketika add ABK
|
ketika add ABK
|
||||||
*/
|
*/
|
||||||
route.Post("/assignment/:idDailyChecklist", ctrl.CreateAssignment)
|
route.Post("/assignment/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.CreateAssignment)
|
||||||
|
|
||||||
// remove assignment
|
// remove assignment
|
||||||
/*
|
/*
|
||||||
ketika remove ABK
|
ketika remove ABK
|
||||||
*/
|
*/
|
||||||
route.Delete("/:idDailyChecklist/assignments/:idEmployee", ctrl.RemoveAssignment)
|
route.Delete("/:idDailyChecklist/assignments/:idEmployee", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.RemoveAssignment)
|
||||||
|
|
||||||
//get all tasks
|
//get all tasks
|
||||||
route.Get("/tasks", ctrl.GetAllTasks)
|
route.Get("/tasks", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.GetAllTasks)
|
||||||
|
|
||||||
// update assignment
|
// update assignment
|
||||||
/*
|
/*
|
||||||
ketika check dan uncheck tugas oleh ABK
|
ketika check dan uncheck tugas oleh ABK
|
||||||
*/
|
*/
|
||||||
route.Post("/assignment", ctrl.UpdateAssignment)
|
route.Post("/assignment", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.UpdateAssignment)
|
||||||
|
|
||||||
route.Patch("/:idDailyChecklist", ctrl.UpdateOne)
|
route.Patch("/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.UpdateOne)
|
||||||
route.Delete("/:idDailyChecklist", ctrl.DeleteOne)
|
route.Delete("/:idDailyChecklist", m.RequirePermissions(m.P_DailyChecklistCreateOne), ctrl.DeleteOne)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,9 +15,9 @@ func ConfigChecklistRoutes(v1 fiber.Router, u user.UserService, s configChecklis
|
|||||||
route := v1.Group("/config-checklists")
|
route := v1.Group("/config-checklists")
|
||||||
route.Use(m.Auth(u))
|
route.Use(m.Auth(u))
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
route.Get("/", m.RequirePermissions(m.P_DailyChecklistActivityConfig), ctrl.GetAll)
|
||||||
route.Post("/", ctrl.CreateOne)
|
route.Post("/", m.RequirePermissions(m.P_DailyChecklistActivityConfig), ctrl.CreateOne)
|
||||||
route.Get("/:id", ctrl.GetOne)
|
route.Get("/:id", m.RequirePermissions(m.P_DailyChecklistActivityConfig), ctrl.GetOne)
|
||||||
route.Patch("/:id", ctrl.UpdateOne)
|
route.Patch("/:id", m.RequirePermissions(m.P_DailyChecklistActivityConfig), ctrl.UpdateOne)
|
||||||
route.Delete("/:id", ctrl.DeleteOne)
|
route.Delete("/:id", m.RequirePermissions(m.P_DailyChecklistActivityConfig), ctrl.DeleteOne)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,9 +15,9 @@ func EmployeesRoutes(v1 fiber.Router, u user.UserService, s employees.EmployeesS
|
|||||||
route := v1.Group("/employees")
|
route := v1.Group("/employees")
|
||||||
route.Use(m.Auth(u))
|
route.Use(m.Auth(u))
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
route.Get("/", m.RequirePermissions(m.P_DailyChecklistEmployee), ctrl.GetAll)
|
||||||
route.Post("/", ctrl.CreateOne)
|
route.Post("/", m.RequirePermissions(m.P_DailyChecklistEmployee), ctrl.CreateOne)
|
||||||
route.Get("/:id", ctrl.GetOne)
|
route.Get("/:id", m.RequirePermissions(m.P_DailyChecklistEmployee), ctrl.GetOne)
|
||||||
route.Patch("/:id", ctrl.UpdateOne)
|
route.Patch("/:id", m.RequirePermissions(m.P_DailyChecklistEmployee), ctrl.UpdateOne)
|
||||||
route.Delete("/:id", ctrl.DeleteOne)
|
route.Delete("/:id", m.RequirePermissions(m.P_DailyChecklistEmployee), ctrl.DeleteOne)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -37,7 +37,6 @@ type NonstockSupplierDTO struct {
|
|||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Alias string `json:"alias"`
|
Alias string `json:"alias"`
|
||||||
Category string `json:"category"`
|
Category string `json:"category"`
|
||||||
Price float64 `json:"price"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// === Mapper Functions ===
|
// === Mapper Functions ===
|
||||||
@@ -121,7 +120,6 @@ func toNonstockSupplierDTOs(relations []entity.NonstockSupplier) []NonstockSuppl
|
|||||||
Name: relation.Supplier.Name,
|
Name: relation.Supplier.Name,
|
||||||
Alias: relation.Supplier.Alias,
|
Alias: relation.Supplier.Alias,
|
||||||
Category: relation.Supplier.Category,
|
Category: relation.Supplier.Category,
|
||||||
Price: relation.Price,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -61,30 +61,20 @@ func (r *NonstockRepositoryImpl) SyncSuppliersDiff(ctx context.Context, tx *gorm
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
existingMap := make(map[uint]entity.NonstockSupplier, len(existing))
|
existingMap := make(map[uint]struct{}, len(existing))
|
||||||
for _, rel := range existing {
|
for _, rel := range existing {
|
||||||
existingMap[rel.SupplierId] = rel
|
existingMap[rel.SupplierId] = struct{}{}
|
||||||
}
|
}
|
||||||
|
|
||||||
incomingMap := make(map[uint]struct{}, len(suppliers))
|
incomingMap := make(map[uint]struct{}, len(suppliers))
|
||||||
for _, rel := range suppliers {
|
for _, rel := range suppliers {
|
||||||
incomingMap[rel.SupplierId] = struct{}{}
|
incomingMap[rel.SupplierId] = struct{}{}
|
||||||
if existingRel, exists := existingMap[rel.SupplierId]; exists {
|
if _, exists := existingMap[rel.SupplierId]; exists {
|
||||||
if existingRel.Price != rel.Price {
|
|
||||||
if err := db.WithContext(ctx).
|
|
||||||
Model(&entity.NonstockSupplier{}).
|
|
||||||
Where("nonstock_id = ? AND supplier_id = ?", nonstockID, rel.SupplierId).
|
|
||||||
Update("price", rel.Price).
|
|
||||||
Error; err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
record := entity.NonstockSupplier{
|
record := entity.NonstockSupplier{
|
||||||
NonstockId: nonstockID,
|
NonstockId: nonstockID,
|
||||||
SupplierId: rel.SupplierId,
|
SupplierId: rel.SupplierId,
|
||||||
Price: rel.Price,
|
|
||||||
}
|
}
|
||||||
if err := db.WithContext(ctx).Create(&record).Error; err != nil {
|
if err := db.WithContext(ctx).Create(&record).Error; err != nil {
|
||||||
return err
|
return err
|
||||||
|
|||||||
@@ -115,19 +115,18 @@ func (s *nonstockService) CreateOne(c *fiber.Ctx, req *validation.Create) (*enti
|
|||||||
supplierLinks []entity.NonstockSupplier
|
supplierLinks []entity.NonstockSupplier
|
||||||
supplierIDs []uint
|
supplierIDs []uint
|
||||||
)
|
)
|
||||||
if len(req.Suppliers) > 0 {
|
if len(req.SupplierIDs) > 0 {
|
||||||
seen := make(map[uint]struct{}, len(req.Suppliers))
|
seen := make(map[uint]struct{}, len(req.SupplierIDs))
|
||||||
supplierLinks = make([]entity.NonstockSupplier, 0, len(req.Suppliers))
|
supplierLinks = make([]entity.NonstockSupplier, 0, len(req.SupplierIDs))
|
||||||
supplierIDs = make([]uint, 0, len(req.Suppliers))
|
supplierIDs = make([]uint, 0, len(req.SupplierIDs))
|
||||||
for _, supplier := range req.Suppliers {
|
for _, supplierID := range req.SupplierIDs {
|
||||||
if _, exists := seen[supplier.SupplierID]; exists {
|
if _, exists := seen[supplierID]; exists {
|
||||||
return nil, fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Duplicate supplier_id %d", supplier.SupplierID))
|
return nil, fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Duplicate supplier_id %d", supplierID))
|
||||||
}
|
}
|
||||||
seen[supplier.SupplierID] = struct{}{}
|
seen[supplierID] = struct{}{}
|
||||||
supplierIDs = append(supplierIDs, supplier.SupplierID)
|
supplierIDs = append(supplierIDs, supplierID)
|
||||||
supplierLinks = append(supplierLinks, entity.NonstockSupplier{
|
supplierLinks = append(supplierLinks, entity.NonstockSupplier{
|
||||||
SupplierId: supplier.SupplierID,
|
SupplierId: supplierID,
|
||||||
Price: supplier.Price,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
supplierList, supplierErr := s.Repository.GetSuppliersByIDs(ctx, supplierIDs)
|
supplierList, supplierErr := s.Repository.GetSuppliersByIDs(ctx, supplierIDs)
|
||||||
@@ -212,21 +211,20 @@ func (s nonstockService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint
|
|||||||
|
|
||||||
var supplierLinks []entity.NonstockSupplier
|
var supplierLinks []entity.NonstockSupplier
|
||||||
var supplierUpdate bool
|
var supplierUpdate bool
|
||||||
if req.Suppliers != nil {
|
if req.SupplierIDs != nil {
|
||||||
supplierUpdate = true
|
supplierUpdate = true
|
||||||
if len(*req.Suppliers) > 0 {
|
if len(*req.SupplierIDs) > 0 {
|
||||||
seen := make(map[uint]struct{}, len(*req.Suppliers))
|
seen := make(map[uint]struct{}, len(*req.SupplierIDs))
|
||||||
supplierLinks = make([]entity.NonstockSupplier, 0, len(*req.Suppliers))
|
supplierLinks = make([]entity.NonstockSupplier, 0, len(*req.SupplierIDs))
|
||||||
supplierIDs := make([]uint, 0, len(*req.Suppliers))
|
supplierIDs := make([]uint, 0, len(*req.SupplierIDs))
|
||||||
for _, supplier := range *req.Suppliers {
|
for _, supplierID := range *req.SupplierIDs {
|
||||||
if _, exists := seen[supplier.SupplierID]; exists {
|
if _, exists := seen[supplierID]; exists {
|
||||||
return nil, fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Duplicate supplier_id %d", supplier.SupplierID))
|
return nil, fiber.NewError(fiber.StatusBadRequest, fmt.Sprintf("Duplicate supplier_id %d", supplierID))
|
||||||
}
|
}
|
||||||
seen[supplier.SupplierID] = struct{}{}
|
seen[supplierID] = struct{}{}
|
||||||
supplierIDs = append(supplierIDs, supplier.SupplierID)
|
supplierIDs = append(supplierIDs, supplierID)
|
||||||
supplierLinks = append(supplierLinks, entity.NonstockSupplier{
|
supplierLinks = append(supplierLinks, entity.NonstockSupplier{
|
||||||
SupplierId: supplier.SupplierID,
|
SupplierId: supplierID,
|
||||||
Price: supplier.Price,
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,22 +1,17 @@
|
|||||||
package validation
|
package validation
|
||||||
|
|
||||||
type SupplierPrice struct {
|
|
||||||
SupplierID uint `json:"supplier_id" validate:"required,gt=0"`
|
|
||||||
Price float64 `json:"price" validate:"required,gte=0"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Create struct {
|
type Create struct {
|
||||||
Name string `json:"name" validate:"required_strict,min=3,max=50"`
|
Name string `json:"name" validate:"required_strict,min=3,max=50"`
|
||||||
UomID uint `json:"uom_id" validate:"required,gt=0"`
|
UomID uint `json:"uom_id" validate:"required,gt=0"`
|
||||||
Suppliers []SupplierPrice `json:"suppliers,omitempty" validate:"omitempty,dive"`
|
SupplierIDs []uint `json:"supplier_ids,omitempty" validate:"omitempty,dive,gt=0"`
|
||||||
Flags []string `json:"flags" validate:"dive,max=50"`
|
Flags []string `json:"flags" validate:"dive,max=50"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Update struct {
|
type Update struct {
|
||||||
Name *string `json:"name,omitempty" validate:"omitempty,min=3,max=50"`
|
Name *string `json:"name,omitempty" validate:"omitempty,min=3,max=50"`
|
||||||
UomID *uint `json:"uom_id,omitempty" validate:"omitempty,gt=0"`
|
UomID *uint `json:"uom_id,omitempty" validate:"omitempty,gt=0"`
|
||||||
Suppliers *[]SupplierPrice `json:"suppliers,omitempty" validate:"omitempty,dive"`
|
SupplierIDs *[]uint `json:"supplier_ids,omitempty" validate:"omitempty,dive,gt=0"`
|
||||||
Flags *[]string `json:"flags,omitempty" validate:"omitempty,dive,max=50"`
|
Flags *[]string `json:"flags,omitempty" validate:"omitempty,dive,max=50"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Query struct {
|
type Query struct {
|
||||||
|
|||||||
@@ -15,9 +15,9 @@ func PhaseActivityRoutes(v1 fiber.Router, u user.UserService, s phaseActivity.Ph
|
|||||||
route := v1.Group("/phase-activities")
|
route := v1.Group("/phase-activities")
|
||||||
route.Use(m.Auth(u))
|
route.Use(m.Auth(u))
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
route.Get("/", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.GetAll)
|
||||||
route.Post("/", ctrl.CreateOne)
|
route.Post("/", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.CreateOne)
|
||||||
route.Get("/:id", ctrl.GetOne)
|
route.Get("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.GetOne)
|
||||||
route.Patch("/:id", ctrl.UpdateOne)
|
route.Patch("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.UpdateOne)
|
||||||
route.Delete("/:id", ctrl.DeleteOne)
|
route.Delete("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.DeleteOne)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,9 +15,9 @@ func PhasesRoutes(v1 fiber.Router, u user.UserService, s phases.PhasesService) {
|
|||||||
route := v1.Group("/phases")
|
route := v1.Group("/phases")
|
||||||
route.Use(m.Auth(u))
|
route.Use(m.Auth(u))
|
||||||
|
|
||||||
route.Get("/", ctrl.GetAll)
|
route.Get("/", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.GetAll)
|
||||||
route.Post("/", ctrl.CreateOne)
|
route.Post("/", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.CreateOne)
|
||||||
route.Get("/:id", ctrl.GetOne)
|
route.Get("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.GetOne)
|
||||||
route.Patch("/:id", ctrl.UpdateOne)
|
route.Patch("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.UpdateOne)
|
||||||
route.Delete("/:id", ctrl.DeleteOne)
|
route.Delete("/:id", m.RequirePermissions(m.P_DailyChecklistActivity), ctrl.DeleteOne)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,7 +10,6 @@ import (
|
|||||||
type SupplierNonstockDTO struct {
|
type SupplierNonstockDTO struct {
|
||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
Price float64 `json:"price"`
|
|
||||||
Uom *uomDTO.UomRelationDTO `json:"uom,omitempty"`
|
Uom *uomDTO.UomRelationDTO `json:"uom,omitempty"`
|
||||||
Flags []string `json:"flags"`
|
Flags []string `json:"flags"`
|
||||||
}
|
}
|
||||||
@@ -43,7 +42,6 @@ func toSupplierNonstockDTOs(relations []entity.NonstockSupplier) []SupplierNonst
|
|||||||
result = append(result, SupplierNonstockDTO{
|
result = append(result, SupplierNonstockDTO{
|
||||||
Id: Nonstock.Id,
|
Id: Nonstock.Id,
|
||||||
Name: Nonstock.Name,
|
Name: Nonstock.Name,
|
||||||
Price: relation.Price,
|
|
||||||
Uom: uomRef,
|
Uom: uomRef,
|
||||||
Flags: flags,
|
Flags: flags,
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
|
warehouseDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/dto"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/dto"
|
"gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/dto"
|
||||||
service "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/services"
|
service "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/services"
|
||||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/validations"
|
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/validations"
|
||||||
@@ -278,14 +279,22 @@ func (u *ProjectflockController) LookupProjectFlockKandang(c *fiber.Ctx) error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
_ = availableStock
|
||||||
|
|
||||||
dtoResult := dto.ToProjectFlockKandangDTO(*result)
|
dtoResult := dto.ToProjectFlockKandangDTO(*result)
|
||||||
dtoResult.AvailableQuantity = float64(availableStock)
|
if population, err := u.ProjectflockService.GetProjectFlockKandangPopulation(c, result.Id); err != nil {
|
||||||
|
return err
|
||||||
|
} else {
|
||||||
|
dtoResult.AvailableQuantity = population
|
||||||
|
}
|
||||||
|
if warehouse, werr := u.ProjectflockService.GetWarehouseByKandangID(c, result.KandangId); werr != nil {
|
||||||
|
return werr
|
||||||
|
} else if warehouse != nil {
|
||||||
|
mapped := warehouseDTO.ToWarehouseRelationDTO(*warehouse)
|
||||||
|
dtoResult.Warehouse = &mapped
|
||||||
|
}
|
||||||
if withPopulation {
|
if withPopulation {
|
||||||
population, err := u.ProjectflockService.GetProjectFlockKandangPopulation(c, result.Id)
|
population := dtoResult.AvailableQuantity
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
dtoResult.Population = &population
|
dtoResult.Population = &population
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import (
|
|||||||
kandangDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/kandangs/dto"
|
kandangDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/kandangs/dto"
|
||||||
locationDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/locations/dto"
|
locationDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/locations/dto"
|
||||||
productionStandardDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/production-standards/dto"
|
productionStandardDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/production-standards/dto"
|
||||||
|
warehouseDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/dto"
|
||||||
userDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/users/dto"
|
userDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/users/dto"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -17,24 +18,26 @@ type KandangWithPivotDTO struct {
|
|||||||
|
|
||||||
type ProjectFlockWithPivotDTO struct {
|
type ProjectFlockWithPivotDTO struct {
|
||||||
ProjectFlockRelationDTO
|
ProjectFlockRelationDTO
|
||||||
Area *areaDTO.AreaRelationDTO `json:"area,omitempty"`
|
Area *areaDTO.AreaRelationDTO `json:"area,omitempty"`
|
||||||
Category string `json:"category"`
|
Category string `json:"category"`
|
||||||
Fcr *fcrDTO.FcrRelationDTO `json:"fcr,omitempty"`
|
Fcr *fcrDTO.FcrRelationDTO `json:"fcr,omitempty"`
|
||||||
ProductionStandard *productionStandardDTO.ProductionStandardRelationDTO `json:"production_standard,omitempty"`
|
ProductionStandard *productionStandardDTO.ProductionStandardRelationDTO `json:"production_standard,omitempty"`
|
||||||
Location *locationDTO.LocationRelationDTO `json:"location,omitempty"`
|
ProductionStandardId uint `json:"production_standard_id"`
|
||||||
Kandangs []KandangWithPivotDTO `json:"kandangs,omitempty"`
|
Location *locationDTO.LocationRelationDTO `json:"location,omitempty"`
|
||||||
CreatedUser *userDTO.UserRelationDTO `json:"created_user,omitempty"`
|
Kandangs []KandangWithPivotDTO `json:"kandangs,omitempty"`
|
||||||
|
CreatedUser *userDTO.UserRelationDTO `json:"created_user,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProjectFlockKandangDTO struct {
|
type ProjectFlockKandangDTO struct {
|
||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
ProjectFlockKandangId uint `json:"project_flock_kandang_id"`
|
ProjectFlockKandangId uint `json:"project_flock_kandang_id"`
|
||||||
ProjectFlockId uint `json:"project_flock_id"`
|
ProjectFlockId uint `json:"project_flock_id"`
|
||||||
KandangId uint `json:"kandang_id"`
|
KandangId uint `json:"kandang_id"`
|
||||||
Kandang *kandangDTO.KandangRelationDTO `json:"kandang,omitempty"`
|
Kandang *kandangDTO.KandangRelationDTO `json:"kandang,omitempty"`
|
||||||
ProjectFlock *ProjectFlockWithPivotDTO `json:"project_flock,omitempty"`
|
Warehouse *warehouseDTO.WarehouseRelationDTO `json:"warehouse,omitempty"`
|
||||||
AvailableQuantity float64 `json:"available_quantity"`
|
ProjectFlock *ProjectFlockWithPivotDTO `json:"project_flock,omitempty"`
|
||||||
Population *float64 `json:"population,omitempty"`
|
AvailableQuantity float64 `json:"available_quantity"`
|
||||||
|
Population *float64 `json:"population,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToProjectFlockKandangDTO(e entity.ProjectFlockKandang) ProjectFlockKandangDTO {
|
func ToProjectFlockKandangDTO(e entity.ProjectFlockKandang) ProjectFlockKandangDTO {
|
||||||
@@ -53,7 +56,8 @@ func ToProjectFlockKandangDTO(e entity.ProjectFlockKandang) ProjectFlockKandangD
|
|||||||
Period: e.Period,
|
Period: e.Period,
|
||||||
FlockName: e.ProjectFlock.FlockName,
|
FlockName: e.ProjectFlock.FlockName,
|
||||||
},
|
},
|
||||||
Category: e.ProjectFlock.Category,
|
Category: e.ProjectFlock.Category,
|
||||||
|
ProductionStandardId: e.ProjectFlock.ProductionStandardId,
|
||||||
}
|
}
|
||||||
|
|
||||||
if e.ProjectFlock.Area.Id != 0 {
|
if e.ProjectFlock.Area.Id != 0 {
|
||||||
|
|||||||
@@ -38,6 +38,7 @@ type ProjectflockService interface {
|
|||||||
GetOne(ctx *fiber.Ctx, id uint) (*entity.ProjectFlock, *flockDTO.FlockRelationDTO, error)
|
GetOne(ctx *fiber.Ctx, id uint) (*entity.ProjectFlock, *flockDTO.FlockRelationDTO, error)
|
||||||
CreateOne(ctx *fiber.Ctx, req *validation.Create) (*entity.ProjectFlock, error)
|
CreateOne(ctx *fiber.Ctx, req *validation.Create) (*entity.ProjectFlock, error)
|
||||||
GetAvailableDocQuantity(ctx *fiber.Ctx, kandangID uint) (float64, error)
|
GetAvailableDocQuantity(ctx *fiber.Ctx, kandangID uint) (float64, error)
|
||||||
|
GetWarehouseByKandangID(ctx *fiber.Ctx, kandangID uint) (*entity.Warehouse, error)
|
||||||
DeleteOne(ctx *fiber.Ctx, id uint) error
|
DeleteOne(ctx *fiber.Ctx, id uint) error
|
||||||
GetProjectFlockKandangByProjectAndKandang(ctx *fiber.Ctx, projectFlockID uint, kandangID uint) (*entity.ProjectFlockKandang, float64, error)
|
GetProjectFlockKandangByProjectAndKandang(ctx *fiber.Ctx, projectFlockID uint, kandangID uint) (*entity.ProjectFlockKandang, float64, error)
|
||||||
GetProjectFlockKandangPopulation(ctx *fiber.Ctx, projectFlockKandangID uint) (float64, error)
|
GetProjectFlockKandangPopulation(ctx *fiber.Ctx, projectFlockKandangID uint) (float64, error)
|
||||||
@@ -518,6 +519,31 @@ func (s projectflockService) GetAvailableDocQuantity(ctx *fiber.Ctx, kandangID u
|
|||||||
return total, nil
|
return total, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s projectflockService) GetWarehouseByKandangID(ctx *fiber.Ctx, kandangID uint) (*entity.Warehouse, error) {
|
||||||
|
if kandangID == 0 || s.WarehouseRepo == nil {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var warehouse entity.Warehouse
|
||||||
|
err := s.WarehouseRepo.DB().WithContext(ctx.Context()).
|
||||||
|
Preload("Area").
|
||||||
|
Preload("Location").
|
||||||
|
Preload("Kandang").
|
||||||
|
Where("kandang_id = ?", kandangID).
|
||||||
|
Where("deleted_at IS NULL").
|
||||||
|
Order("id DESC").
|
||||||
|
First(&warehouse).Error
|
||||||
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to fetch warehouse for kandang %d: %+v", kandangID, err)
|
||||||
|
return nil, fiber.NewError(fiber.StatusInternalServerError, "Failed to fetch warehouse")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &warehouse, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (s projectflockService) GetProjectPeriods(c *fiber.Ctx, projectIDs []uint) (map[uint]int, error) {
|
func (s projectflockService) GetProjectPeriods(c *fiber.Ctx, projectIDs []uint) (map[uint]int, error) {
|
||||||
if len(projectIDs) == 0 {
|
if len(projectIDs) == 0 {
|
||||||
return map[uint]int{}, nil
|
return map[uint]int{}, nil
|
||||||
|
|||||||
@@ -82,11 +82,11 @@ type RecordingListDTO struct {
|
|||||||
CreatedUser *userDTO.UserRelationDTO `json:"created_user,omitempty"`
|
CreatedUser *userDTO.UserRelationDTO `json:"created_user,omitempty"`
|
||||||
CreatedAt time.Time `json:"created_at"`
|
CreatedAt time.Time `json:"created_at"`
|
||||||
UpdatedAt time.Time `json:"updated_at"`
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Warehouse *RecordingWarehouseDTO `json:"warehouse,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type RecordingDetailDTO struct {
|
type RecordingDetailDTO struct {
|
||||||
RecordingListDTO
|
RecordingListDTO
|
||||||
Warehouse *RecordingWarehouseDTO `json:"warehouse,omitempty"`
|
|
||||||
ProductCategory string `json:"product_category"`
|
ProductCategory string `json:"product_category"`
|
||||||
Depletions []RecordingDepletionDTO `json:"depletions"`
|
Depletions []RecordingDepletionDTO `json:"depletions"`
|
||||||
Stocks []RecordingStockDTO `json:"stocks"`
|
Stocks []RecordingStockDTO `json:"stocks"`
|
||||||
@@ -133,7 +133,6 @@ func ToRecordingDetailDTO(e entity.Recording) RecordingDetailDTO {
|
|||||||
|
|
||||||
return RecordingDetailDTO{
|
return RecordingDetailDTO{
|
||||||
RecordingListDTO: listDTO,
|
RecordingListDTO: listDTO,
|
||||||
Warehouse: recordingWarehouseDTO(e),
|
|
||||||
ProductCategory: recordingProductCategory(e),
|
ProductCategory: recordingProductCategory(e),
|
||||||
Depletions: ToRecordingDepletionDTOs(e.Depletions),
|
Depletions: ToRecordingDepletionDTOs(e.Depletions),
|
||||||
Stocks: ToRecordingStockDTOs(e.Stocks),
|
Stocks: ToRecordingStockDTOs(e.Stocks),
|
||||||
@@ -203,6 +202,7 @@ func toRecordingListDTO(e entity.Recording) RecordingListDTO {
|
|||||||
CreatedAt: e.CreatedAt,
|
CreatedAt: e.CreatedAt,
|
||||||
UpdatedAt: e.UpdatedAt,
|
UpdatedAt: e.UpdatedAt,
|
||||||
CreatedUser: createdUser,
|
CreatedUser: createdUser,
|
||||||
|
Warehouse: recordingWarehouseDTO(e),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -43,6 +43,22 @@ func (RecordingModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate
|
|||||||
)
|
)
|
||||||
|
|
||||||
fifoService := commonSvc.NewFifoService(db, stockAllocationRepo, productWarehouseRepo, utils.Log)
|
fifoService := commonSvc.NewFifoService(db, stockAllocationRepo, productWarehouseRepo, utils.Log)
|
||||||
|
if err := fifoService.RegisterStockable(fifo.StockableConfig{
|
||||||
|
Key: fifo.StockableKeyRecordingEgg,
|
||||||
|
Table: "recording_eggs",
|
||||||
|
Columns: fifo.StockableColumns{
|
||||||
|
ID: "id",
|
||||||
|
ProductWarehouseID: "product_warehouse_id",
|
||||||
|
TotalQuantity: "total_qty",
|
||||||
|
TotalUsedQuantity: "total_used",
|
||||||
|
CreatedAt: "created_at",
|
||||||
|
},
|
||||||
|
OrderBy: []string{"created_at ASC", "id ASC"},
|
||||||
|
}); err != nil {
|
||||||
|
if !strings.Contains(strings.ToLower(err.Error()), "already registered") {
|
||||||
|
panic(fmt.Sprintf("failed to register recording egg stockable workflow: %v", err))
|
||||||
|
}
|
||||||
|
}
|
||||||
if err := fifoService.RegisterUsable(fifo.UsableConfig{
|
if err := fifoService.RegisterUsable(fifo.UsableConfig{
|
||||||
Key: fifo.UsableKeyRecordingStock,
|
Key: fifo.UsableKeyRecordingStock,
|
||||||
Table: "recording_stocks",
|
Table: "recording_stocks",
|
||||||
|
|||||||
@@ -290,8 +290,19 @@ func (s *recordingService) CreateOne(c *fiber.Ctx, req *validation.Create) (*ent
|
|||||||
s.Log.Errorf("Failed to persist eggs: %+v", err)
|
s.Log.Errorf("Failed to persist eggs: %+v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
if s.FifoSvc != nil {
|
||||||
|
if err := s.replenishRecordingEggs(ctx, tx, mappedEggs); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if err := s.adjustProductWarehouseQuantities(ctx, tx, buildWarehouseDeltas(nil, mappedDepletions, nil, mappedEggs)); err != nil {
|
var warehouseDeltas map[uint]float64
|
||||||
|
if s.FifoSvc != nil {
|
||||||
|
warehouseDeltas = buildWarehouseDeltas(nil, mappedDepletions, nil, nil)
|
||||||
|
} else {
|
||||||
|
warehouseDeltas = buildWarehouseDeltas(nil, mappedDepletions, nil, mappedEggs)
|
||||||
|
}
|
||||||
|
if err := s.adjustProductWarehouseQuantities(ctx, tx, warehouseDeltas); err != nil {
|
||||||
s.Log.Errorf("Failed to adjust product warehouses: %+v", err)
|
s.Log.Errorf("Failed to adjust product warehouses: %+v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -438,6 +449,16 @@ func (s recordingService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uin
|
|||||||
}
|
}
|
||||||
|
|
||||||
if hasEggChanges {
|
if hasEggChanges {
|
||||||
|
if s.FifoSvc != nil {
|
||||||
|
if err := ensureRecordingEggsUnused(existingEggs); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := s.adjustProductWarehouseQuantities(ctx, tx, buildWarehouseDeltas(nil, nil, existingEggs, nil)); err != nil {
|
||||||
|
s.Log.Errorf("Failed to adjust product warehouses for eggs: %+v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if err := s.Repository.DeleteEggs(tx, recordingEntity.Id); err != nil {
|
if err := s.Repository.DeleteEggs(tx, recordingEntity.Id); err != nil {
|
||||||
s.Log.Errorf("Failed to clear eggs: %+v", err)
|
s.Log.Errorf("Failed to clear eggs: %+v", err)
|
||||||
return err
|
return err
|
||||||
@@ -449,9 +470,15 @@ func (s recordingService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uin
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.adjustProductWarehouseQuantities(ctx, tx, buildWarehouseDeltas(nil, nil, existingEggs, mappedEggs)); err != nil {
|
if s.FifoSvc != nil {
|
||||||
s.Log.Errorf("Failed to adjust product warehouses for eggs: %+v", err)
|
if err := s.replenishRecordingEggs(ctx, tx, mappedEggs); err != nil {
|
||||||
return err
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if err := s.adjustProductWarehouseQuantities(ctx, tx, buildWarehouseDeltas(nil, nil, existingEggs, mappedEggs)); err != nil {
|
||||||
|
s.Log.Errorf("Failed to adjust product warehouses for eggs: %+v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -626,6 +653,11 @@ func (s recordingService) DeleteOne(c *fiber.Ctx, id uint) error {
|
|||||||
s.Log.Errorf("Failed to list eggs before delete: %+v", err)
|
s.Log.Errorf("Failed to list eggs before delete: %+v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
if s.FifoSvc != nil {
|
||||||
|
if err := ensureRecordingEggsUnused(oldEggs); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
oldStocks, err := s.Repository.ListStocks(tx, id)
|
oldStocks, err := s.Repository.ListStocks(tx, id)
|
||||||
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
|
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
@@ -802,6 +834,32 @@ func (s *recordingService) adjustProductWarehouseQuantities(ctx context.Context,
|
|||||||
return s.ProductWarehouseRepo.AdjustQuantities(ctx, deltas, func(*gorm.DB) *gorm.DB { return tx })
|
return s.ProductWarehouseRepo.AdjustQuantities(ctx, deltas, func(*gorm.DB) *gorm.DB { return tx })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *recordingService) replenishRecordingEggs(ctx context.Context, tx *gorm.DB, eggs []entity.RecordingEgg) error {
|
||||||
|
if len(eggs) == 0 || s.FifoSvc == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, egg := range eggs {
|
||||||
|
if egg.Id == 0 || egg.ProductWarehouseId == 0 || egg.Qty <= 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
note := fmt.Sprintf("Recording egg #%d", egg.Id)
|
||||||
|
if _, err := s.FifoSvc.Replenish(ctx, commonSvc.StockReplenishRequest{
|
||||||
|
StockableKey: fifo.StockableKeyRecordingEgg,
|
||||||
|
StockableID: egg.Id,
|
||||||
|
ProductWarehouseID: egg.ProductWarehouseId,
|
||||||
|
Quantity: float64(egg.Qty),
|
||||||
|
Note: ¬e,
|
||||||
|
Tx: tx,
|
||||||
|
}); err != nil {
|
||||||
|
s.Log.Errorf("Failed to replenish FIFO stock for recording egg %d: %+v", egg.Id, err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
type desiredStock struct {
|
type desiredStock struct {
|
||||||
Usage float64
|
Usage float64
|
||||||
Pending float64
|
Pending float64
|
||||||
@@ -922,6 +980,14 @@ type eggTotals struct {
|
|||||||
Weight float64
|
Weight float64
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func ensureRecordingEggsUnused(eggs []entity.RecordingEgg) error {
|
||||||
|
for _, egg := range eggs {
|
||||||
|
if egg.TotalUsed > 0 {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, "Recording egg sudah digunakan sehingga tidak dapat diubah")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func stocksMatch(existing []entity.RecordingStock, incoming []validation.Stock) bool {
|
func stocksMatch(existing []entity.RecordingStock, incoming []validation.Stock) bool {
|
||||||
hasPending := false
|
hasPending := false
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ import (
|
|||||||
|
|
||||||
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
expenseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/expenses/repositories"
|
||||||
marketingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/marketing/repositories"
|
marketingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/marketing/repositories"
|
||||||
|
productionStandardRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/production-standards/repositories"
|
||||||
chickinRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
chickinRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
||||||
recordingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/repositories"
|
recordingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/repositories"
|
||||||
purchaseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/purchases/repositories"
|
purchaseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/purchases/repositories"
|
||||||
@@ -34,10 +35,26 @@ func (RepportModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *
|
|||||||
debtSupplierRepository := repportRepo.NewDebtSupplierRepository(db)
|
debtSupplierRepository := repportRepo.NewDebtSupplierRepository(db)
|
||||||
hppPerKandangRepository := repportRepo.NewHppPerKandangRepository(db)
|
hppPerKandangRepository := repportRepo.NewHppPerKandangRepository(db)
|
||||||
productionResultRepository := repportRepo.NewProductionResultRepository(db)
|
productionResultRepository := repportRepo.NewProductionResultRepository(db)
|
||||||
|
standardGrowthDetailRepository := productionStandardRepo.NewStandardGrowthDetailRepository(db)
|
||||||
|
productionStandardDetailRepository := productionStandardRepo.NewProductionStandardDetailRepository(db)
|
||||||
userRepository := rUser.NewUserRepository(db)
|
userRepository := rUser.NewUserRepository(db)
|
||||||
|
|
||||||
approvalSvc := approvalService.NewApprovalService(approvalRepository)
|
approvalSvc := approvalService.NewApprovalService(approvalRepository)
|
||||||
repportService := sRepport.NewRepportService(validate, expenseRealizationRepository, marketingDeliveryProductRepository, purchaseRepository, chickinRepository, recordingRepository, approvalSvc, purchaseSupplierRepository, debtSupplierRepository, hppPerKandangRepository, productionResultRepository)
|
repportService := sRepport.NewRepportService(
|
||||||
|
validate,
|
||||||
|
expenseRealizationRepository,
|
||||||
|
marketingDeliveryProductRepository,
|
||||||
|
purchaseRepository,
|
||||||
|
chickinRepository,
|
||||||
|
recordingRepository,
|
||||||
|
approvalSvc,
|
||||||
|
purchaseSupplierRepository,
|
||||||
|
debtSupplierRepository,
|
||||||
|
hppPerKandangRepository,
|
||||||
|
productionResultRepository,
|
||||||
|
standardGrowthDetailRepository,
|
||||||
|
productionStandardDetailRepository,
|
||||||
|
)
|
||||||
userService := sUser.NewUserService(userRepository, validate)
|
userService := sUser.NewUserService(userRepository, validate)
|
||||||
|
|
||||||
RepportRoutes(router, userService, repportService)
|
RepportRoutes(router, userService, repportService)
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import (
|
|||||||
|
|
||||||
type ProductionResultRepository interface {
|
type ProductionResultRepository interface {
|
||||||
GetRecordingsByProjectFlockKandang(ctx context.Context, projectFlockKandangID uint, offset, limit int) ([]entity.Recording, int64, error)
|
GetRecordingsByProjectFlockKandang(ctx context.Context, projectFlockKandangID uint, offset, limit int) ([]entity.Recording, int64, error)
|
||||||
|
GetProductionStandardIDByProjectFlockKandangID(ctx context.Context, projectFlockKandangID uint) (uint, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type productionResultRepositoryImpl struct {
|
type productionResultRepositoryImpl struct {
|
||||||
@@ -76,3 +77,25 @@ func (r *productionResultRepositoryImpl) GetRecordingsByProjectFlockKandang(
|
|||||||
|
|
||||||
return recordings, total, nil
|
return recordings, total, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *productionResultRepositoryImpl) GetProductionStandardIDByProjectFlockKandangID(ctx context.Context, projectFlockKandangID uint) (uint, error) {
|
||||||
|
if projectFlockKandangID == 0 {
|
||||||
|
return 0, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var row struct {
|
||||||
|
ProductionStandardID uint `gorm:"column:production_standard_id"`
|
||||||
|
}
|
||||||
|
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Table("project_flock_kandangs pfk").
|
||||||
|
Select("pf.production_standard_id").
|
||||||
|
Joins("JOIN project_flocks pf ON pf.id = pfk.project_flock_id").
|
||||||
|
Where("pfk.id = ?", projectFlockKandangID).
|
||||||
|
Take(&row).Error
|
||||||
|
if err != nil {
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return row.ProductionStandardID, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package service
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
"math"
|
||||||
"sort"
|
"sort"
|
||||||
@@ -21,6 +22,7 @@ import (
|
|||||||
areaDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/areas/dto"
|
areaDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/areas/dto"
|
||||||
supplierDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/dto"
|
supplierDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/dto"
|
||||||
warehouseDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/dto"
|
warehouseDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/dto"
|
||||||
|
productionStandardRepository "gitlab.com/mbugroup/lti-api.git/internal/modules/master/production-standards/repositories"
|
||||||
chickinRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
chickinRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
||||||
recordingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/repositories"
|
recordingRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings/repositories"
|
||||||
purchaseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/purchases/repositories"
|
purchaseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/purchases/repositories"
|
||||||
@@ -55,6 +57,8 @@ type repportService struct {
|
|||||||
DebtSupplierRepo repportRepo.DebtSupplierRepository
|
DebtSupplierRepo repportRepo.DebtSupplierRepository
|
||||||
HppPerKandangRepo repportRepo.HppPerKandangRepository
|
HppPerKandangRepo repportRepo.HppPerKandangRepository
|
||||||
ProductionResultRepo repportRepo.ProductionResultRepository
|
ProductionResultRepo repportRepo.ProductionResultRepository
|
||||||
|
StandardGrowthDetailRepo productionStandardRepository.StandardGrowthDetailRepository
|
||||||
|
ProductionStandardDetailRepo productionStandardRepository.ProductionStandardDetailRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
type HppCostAggregate struct {
|
type HppCostAggregate struct {
|
||||||
@@ -78,6 +82,8 @@ func NewRepportService(
|
|||||||
debtSupplierRepo repportRepo.DebtSupplierRepository,
|
debtSupplierRepo repportRepo.DebtSupplierRepository,
|
||||||
hppPerKandangRepo repportRepo.HppPerKandangRepository,
|
hppPerKandangRepo repportRepo.HppPerKandangRepository,
|
||||||
productionResultRepo repportRepo.ProductionResultRepository,
|
productionResultRepo repportRepo.ProductionResultRepository,
|
||||||
|
standardGrowthDetailRepo productionStandardRepository.StandardGrowthDetailRepository,
|
||||||
|
productionStandardDetailRepo productionStandardRepository.ProductionStandardDetailRepository,
|
||||||
) RepportService {
|
) RepportService {
|
||||||
return &repportService{
|
return &repportService{
|
||||||
Log: utils.Log,
|
Log: utils.Log,
|
||||||
@@ -92,6 +98,8 @@ func NewRepportService(
|
|||||||
DebtSupplierRepo: debtSupplierRepo,
|
DebtSupplierRepo: debtSupplierRepo,
|
||||||
HppPerKandangRepo: hppPerKandangRepo,
|
HppPerKandangRepo: hppPerKandangRepo,
|
||||||
ProductionResultRepo: productionResultRepo,
|
ProductionResultRepo: productionResultRepo,
|
||||||
|
StandardGrowthDetailRepo: standardGrowthDetailRepo,
|
||||||
|
ProductionStandardDetailRepo: productionStandardDetailRepo,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -285,6 +293,21 @@ func (s *repportService) GetProductionResult(ctx *fiber.Ctx, params *validation.
|
|||||||
|
|
||||||
weeklyResults := summarizeProductionResults(dailyResults, recordsPerWeek)
|
weeklyResults := summarizeProductionResults(dailyResults, recordsPerWeek)
|
||||||
|
|
||||||
|
var productionStandardID uint
|
||||||
|
if s.ProductionResultRepo != nil {
|
||||||
|
standardID, err := s.ProductionResultRepo.GetProductionStandardIDByProjectFlockKandangID(ctx.Context(), params.ProjectFlockKandangID)
|
||||||
|
if err != nil {
|
||||||
|
if !errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, 0, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
productionStandardID = standardID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
standardDetailCache := make(map[int]*entity.ProductionStandardDetail)
|
||||||
|
growthDetailCache := make(map[int]*entity.StandardGrowthDetail)
|
||||||
|
|
||||||
var cumulativeButir int64
|
var cumulativeButir int64
|
||||||
var cumulativeKg float64
|
var cumulativeKg float64
|
||||||
for i := range weeklyResults {
|
for i := range weeklyResults {
|
||||||
@@ -300,6 +323,66 @@ func (s *repportService) GetProductionResult(ctx *fiber.Ctx, params *validation.
|
|||||||
|
|
||||||
cumulativeKg += weeklyResults[i].KgJumlah
|
cumulativeKg += weeklyResults[i].KgJumlah
|
||||||
weeklyResults[i].TotalKg = cumulativeKg
|
weeklyResults[i].TotalKg = cumulativeKg
|
||||||
|
|
||||||
|
if productionStandardID == 0 {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
week := int(weeklyResults[i].Woa)
|
||||||
|
if s.ProductionStandardDetailRepo != nil {
|
||||||
|
detail, ok := standardDetailCache[week]
|
||||||
|
if !ok {
|
||||||
|
fetched, fetchErr := s.ProductionStandardDetailRepo.GetByStandardIDAndWeek(ctx.Context(), productionStandardID, week)
|
||||||
|
if fetchErr != nil {
|
||||||
|
if !errors.Is(fetchErr, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, 0, fetchErr
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
detail = fetched
|
||||||
|
}
|
||||||
|
standardDetailCache[week] = detail
|
||||||
|
}
|
||||||
|
|
||||||
|
if detail != nil {
|
||||||
|
if detail.TargetHenDayProduction != nil {
|
||||||
|
weeklyResults[i].HdStd = *detail.TargetHenDayProduction
|
||||||
|
}
|
||||||
|
if detail.TargetHenHouseProduction != nil {
|
||||||
|
weeklyResults[i].HhStd = *detail.TargetHenHouseProduction
|
||||||
|
}
|
||||||
|
if detail.TargetEggWeight != nil {
|
||||||
|
weeklyResults[i].EwStd = *detail.TargetEggWeight
|
||||||
|
}
|
||||||
|
if detail.TargetEggMass != nil {
|
||||||
|
weeklyResults[i].EmStd = *detail.TargetEggMass
|
||||||
|
}
|
||||||
|
if detail.StandardFCR != nil {
|
||||||
|
weeklyResults[i].FcrStd = *detail.StandardFCR
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if s.StandardGrowthDetailRepo != nil {
|
||||||
|
detail, ok := growthDetailCache[week]
|
||||||
|
if !ok {
|
||||||
|
fetched, fetchErr := s.StandardGrowthDetailRepo.GetByStandardIDAndWeek(ctx.Context(), productionStandardID, week)
|
||||||
|
if fetchErr != nil {
|
||||||
|
if !errors.Is(fetchErr, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, 0, fetchErr
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
detail = fetched
|
||||||
|
}
|
||||||
|
growthDetailCache[week] = detail
|
||||||
|
}
|
||||||
|
|
||||||
|
if detail != nil && detail.FeedIntake != nil {
|
||||||
|
weeklyResults[i].FiStd = *detail.FeedIntake
|
||||||
|
}
|
||||||
|
if detail != nil && detail.TargetMeanBw != nil {
|
||||||
|
weeklyResults[i].StdBw = *detail.TargetMeanBw
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
totalWeeks := int64(math.Ceil(float64(totalRecordings) / float64(recordsPerWeek)))
|
totalWeeks := int64(math.Ceil(float64(totalRecordings) / float64(recordsPerWeek)))
|
||||||
@@ -314,17 +397,17 @@ func mapRecordingToProductionResultDTO(record entity.Recording) dto.ProductionRe
|
|||||||
StdUniformity: "90% up",
|
StdUniformity: "90% up",
|
||||||
DepKum: valueOrZero(record.CumDepletionRate),
|
DepKum: valueOrZero(record.CumDepletionRate),
|
||||||
DepStd: valueOrZero(record.TotalDepletionQty),
|
DepStd: valueOrZero(record.TotalDepletionQty),
|
||||||
|
Hd: valueOrZero(record.HenDay),
|
||||||
|
Fi: valueOrZero(record.FeedIntake),
|
||||||
Fcr: valueOrZero(record.FcrValue),
|
Fcr: valueOrZero(record.FcrValue),
|
||||||
Hh: valueOrZero(record.TotalChickQty),
|
Hh: valueOrZero(record.HenHouse),
|
||||||
|
Em: valueOrZero(record.EggMass),
|
||||||
|
Ew: valueOrZero(record.EggWeight),
|
||||||
}
|
}
|
||||||
|
|
||||||
if record.Day != nil {
|
if record.Day != nil {
|
||||||
result.Woa = float64(*record.Day)
|
result.Woa = float64(*record.Day)
|
||||||
}
|
}
|
||||||
if record.CumIntake != nil {
|
|
||||||
result.Fi = float64(*record.CumIntake)
|
|
||||||
}
|
|
||||||
|
|
||||||
// avgWeight := calculateAverageBodyWeight(record.BodyWeights)
|
// avgWeight := calculateAverageBodyWeight(record.BodyWeights)
|
||||||
avgWeight := 1.0
|
avgWeight := 1.0
|
||||||
if avgWeight > 0 {
|
if avgWeight > 0 {
|
||||||
@@ -351,8 +434,6 @@ func mapRecordingToProductionResultDTO(record entity.Recording) dto.ProductionRe
|
|||||||
result.PersenPutih = roundFloat((float64(result.ButiranPutih)/total)*100, 2)
|
result.PersenPutih = roundFloat((float64(result.ButiranPutih)/total)*100, 2)
|
||||||
result.PersenRetak = roundFloat((float64(result.ButiranRetak)/total)*100, 2)
|
result.PersenRetak = roundFloat((float64(result.ButiranRetak)/total)*100, 2)
|
||||||
result.PersenPecah = roundFloat((float64(result.ButiranPecah)/total)*100, 2)
|
result.PersenPecah = roundFloat((float64(result.ButiranPecah)/total)*100, 2)
|
||||||
result.Ew = (eggSummary.TotalKg * 1000) / total
|
|
||||||
result.Em = eggSummary.TotalKg
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
@@ -464,13 +545,13 @@ func summarizeProductionResults(daily []dto.ProductionResultDTO, groupSize int)
|
|||||||
if end > len(daily) {
|
if end > len(daily) {
|
||||||
end = len(daily)
|
end = len(daily)
|
||||||
}
|
}
|
||||||
result = append(result, aggregateProductionResultGroup(daily[i:end]))
|
result = append(result, aggregateProductionResultGroup(daily[i:end], groupSize))
|
||||||
}
|
}
|
||||||
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
func aggregateProductionResultGroup(group []dto.ProductionResultDTO) dto.ProductionResultDTO {
|
func aggregateProductionResultGroup(group []dto.ProductionResultDTO, groupSize int) dto.ProductionResultDTO {
|
||||||
count := len(group)
|
count := len(group)
|
||||||
if count == 0 {
|
if count == 0 {
|
||||||
return dto.ProductionResultDTO{}
|
return dto.ProductionResultDTO{}
|
||||||
@@ -542,6 +623,10 @@ func aggregateProductionResultGroup(group []dto.ProductionResultDTO) dto.Product
|
|||||||
if divider == 0 {
|
if divider == 0 {
|
||||||
divider = 1
|
divider = 1
|
||||||
}
|
}
|
||||||
|
weeklyDivider := float64(groupSize)
|
||||||
|
if weeklyDivider == 0 {
|
||||||
|
weeklyDivider = divider
|
||||||
|
}
|
||||||
|
|
||||||
agg.Bw = sumBw / divider
|
agg.Bw = sumBw / divider
|
||||||
agg.StdBw = sumStdBw / divider
|
agg.StdBw = sumStdBw / divider
|
||||||
@@ -570,17 +655,17 @@ func aggregateProductionResultGroup(group []dto.ProductionResultDTO) dto.Product
|
|||||||
agg.PersenPecah = roundFloat(sumPersenPecah/percentDivider, 2)
|
agg.PersenPecah = roundFloat(sumPersenPecah/percentDivider, 2)
|
||||||
}
|
}
|
||||||
|
|
||||||
agg.Hd = sumHd / divider
|
agg.Hd = roundFloat(sumHd/weeklyDivider, 2)
|
||||||
agg.HdStd = sumHdStd / divider
|
agg.HdStd = sumHdStd / divider
|
||||||
agg.Fi = sumFi / divider
|
agg.Fi = roundFloat(sumFi/weeklyDivider, 2)
|
||||||
agg.FiStd = sumFiStd / divider
|
agg.FiStd = sumFiStd / divider
|
||||||
agg.Em = sumEm / divider
|
agg.Em = group[count-1].Em
|
||||||
agg.EmStd = sumEmStd / divider
|
agg.EmStd = sumEmStd / divider
|
||||||
agg.Ew = sumEw / divider
|
agg.Ew = group[count-1].Ew
|
||||||
agg.EwStd = sumEwStd / divider
|
agg.EwStd = sumEwStd / divider
|
||||||
agg.Fcr = sumFcr / divider
|
agg.Fcr = roundFloat(sumFcr/weeklyDivider, 2)
|
||||||
agg.FcrStd = sumFcrStd / divider
|
agg.FcrStd = sumFcrStd / divider
|
||||||
agg.Hh = sumHh / divider
|
agg.Hh = roundFloat(sumHh/weeklyDivider, 2)
|
||||||
agg.HhStd = sumHhStd / divider
|
agg.HhStd = sumHhStd / divider
|
||||||
|
|
||||||
return agg
|
return agg
|
||||||
|
|||||||
@@ -15,4 +15,5 @@ const (
|
|||||||
StockableKeyAdjustmentIn StockableKey = "ADJUSTMENT_IN"
|
StockableKeyAdjustmentIn StockableKey = "ADJUSTMENT_IN"
|
||||||
StockableKeyPurchaseItems StockableKey = "PURCHASE_ITEMS"
|
StockableKeyPurchaseItems StockableKey = "PURCHASE_ITEMS"
|
||||||
StockableKeyProjectFlockPopulation StockableKey = "PROJECT_FLOCK_POPULATION"
|
StockableKeyProjectFlockPopulation StockableKey = "PROJECT_FLOCK_POPULATION"
|
||||||
|
StockableKeyRecordingEgg StockableKey = "RECORDING_EGG"
|
||||||
)
|
)
|
||||||
|
|||||||
Reference in New Issue
Block a user