mirror of
https://gitlab.com/mbugroup/lti-api.git
synced 2026-05-24 15:25:43 +00:00
feat(BE-115,116,117): implement chickin CRUD, approve logic, and stock availabilit
This commit is contained in:
@@ -1,18 +0,0 @@
|
|||||||
CREATE TABLE project_chick_ins (
|
|
||||||
id BIGSERIAL PRIMARY KEY,
|
|
||||||
project_floc_id BIGINT NOT NULL REFERENCES project_flocs (id),
|
|
||||||
product_warehouse_id BIGINT NOT NULL REFERENCES product_warehouses (id),
|
|
||||||
chick_in_date DATE NOT NULL,
|
|
||||||
quantity NUMERIC(15, 3) NOT NULL CHECK (quantity > 0),
|
|
||||||
note TEXT,
|
|
||||||
created_by BIGINT NOT NULL REFERENCES users (id),
|
|
||||||
created_at TIMESTAMPTZ DEFAULT now(),
|
|
||||||
updated_at TIMESTAMPTZ DEFAULT now(),
|
|
||||||
deleted_at TIMESTAMPTZ
|
|
||||||
);
|
|
||||||
|
|
||||||
CREATE INDEX idx_project_chick_ins_project_floc_id ON project_chick_ins (project_floc_id);
|
|
||||||
|
|
||||||
CREATE INDEX idx_project_chick_ins_product_warehouse_id ON project_chick_ins (product_warehouse_id);
|
|
||||||
|
|
||||||
CREATE INDEX idx_project_chick_ins_created_by ON project_chick_ins (created_by);
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
DROP TABLE IF EXISTS project_chick_ins;
|
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS project_chickins;
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
CREATE TABLE project_chickins (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
project_floc_id BIGINT NOT NULL,
|
||||||
|
chick_in_date DATE NOT NULL,
|
||||||
|
quantity NUMERIC(15, 3) NOT NULL,
|
||||||
|
note TEXT,
|
||||||
|
created_by BIGINT NOT NULL,
|
||||||
|
created_at TIMESTAMPTZ DEFAULT now(),
|
||||||
|
updated_at TIMESTAMPTZ DEFAULT now(),
|
||||||
|
deleted_at TIMESTAMPTZ
|
||||||
|
);
|
||||||
|
|
||||||
|
CREATE INDEX idx_project_chickins_project_floc_id ON project_chickins (project_floc_id);
|
||||||
|
|
||||||
|
CREATE INDEX idx_project_chickins_created_by ON project_chickins (created_by);
|
||||||
|
|
||||||
|
ALTER TABLE project_chickins
|
||||||
|
ADD CONSTRAINT fk_project_floc_id FOREIGN KEY (project_floc_id) REFERENCES project_flocks (id);
|
||||||
|
|
||||||
|
ALTER TABLE project_chickins
|
||||||
|
ADD CONSTRAINT fk_created_by FOREIGN KEY (created_by) REFERENCES users (id);
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS stock_availabilities;
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
CREATE TABLE stock_availabilities (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
entity_type VARCHAR(50) NOT NULL,
|
||||||
|
entity_id BIGINT NOT NULL,
|
||||||
|
product_id BIGINT,
|
||||||
|
quantity NUMERIC(15, 3) NOT NULL DEFAULT 0,
|
||||||
|
reserved_quantity NUMERIC(15, 3) NOT NULL DEFAULT 0,
|
||||||
|
unit VARCHAR(20),
|
||||||
|
last_updated TIMESTAMPTZ DEFAULT now(),
|
||||||
|
created_at TIMESTAMPTZ DEFAULT now(),
|
||||||
|
deleted_at TIMESTAMPTZ
|
||||||
|
);
|
||||||
|
|
||||||
|
ALTER TABLE stock_availabilities
|
||||||
|
ADD CONSTRAINT fk_product_id FOREIGN KEY (product_id) REFERENCES products (id);
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
DROP TABLE IF EXISTS audit_logs;
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
CREATE TABLE audit_logs (
|
||||||
|
id BIGSERIAL PRIMARY KEY,
|
||||||
|
table_name VARCHAR(100) NOT NULL,
|
||||||
|
record_id BIGINT NOT NULL,
|
||||||
|
action VARCHAR(30) NOT NULL,
|
||||||
|
before_data JSONB,
|
||||||
|
after_data JSONB,
|
||||||
|
changed_by BIGINT,
|
||||||
|
created_at TIMESTAMPTZ DEFAULT now()
|
||||||
|
);
|
||||||
|
|
||||||
|
ALTER TABLE audit_logs
|
||||||
|
ADD CONSTRAINT fk_changed_by FOREIGN KEY (changed_by) REFERENCES users (id);
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
package entities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AuditLog struct {
|
||||||
|
Id uint `gorm:"primaryKey"`
|
||||||
|
TableName string `gorm:"size:100;not null"`
|
||||||
|
RecordId uint `gorm:"not null"`
|
||||||
|
Action string `gorm:"size:30;not null"`
|
||||||
|
BeforeData string `gorm:"type:jsonb"`
|
||||||
|
AfterData string `gorm:"type:jsonb"`
|
||||||
|
ChangedBy uint `gorm:"not null"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||||
|
|
||||||
|
User *User `gorm:"foreignKey:ChangedBy;references:Id"`
|
||||||
|
}
|
||||||
@@ -6,19 +6,19 @@ import (
|
|||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ProjectChickin struct {
|
const ()
|
||||||
Id uint `gorm:"primaryKey"`
|
|
||||||
ProjectFlocId uint `gorm:"not null"`
|
|
||||||
ProductWarehouseId uint `gorm:"not null"`
|
|
||||||
ChickInDate time.Time `gorm:"not null"`
|
|
||||||
Quantity float64 `gorm:"not null"`
|
|
||||||
Note string `gorm:"type:text"`
|
|
||||||
CreatedBy uint `gorm:"not null"`
|
|
||||||
CreatedAt time.Time `gorm:"autoCreateTime"`
|
|
||||||
UpdatedAt time.Time `gorm:"autoUpdateTime"`
|
|
||||||
DeletedAt gorm.DeletedAt `gorm:"index" json:"-"`
|
|
||||||
|
|
||||||
ProjectFloc ProjectFlock `gorm:"foreignKey:ProjectFlocId;references:Id"`
|
type ProjectChickin struct {
|
||||||
ProductWarehouse ProductWarehouse `gorm:"foreignKey:ProductWarehouseId;references:Id"`
|
Id uint `gorm:"primaryKey"`
|
||||||
CreatedUser User `gorm:"foreignKey:CreatedBy;references:Id"`
|
ProjectFlocId uint `gorm:"not null"`
|
||||||
|
ChickInDate time.Time `gorm:"not null"`
|
||||||
|
Quantity float64 `gorm:"not null"`
|
||||||
|
Note string `gorm:"type:text"`
|
||||||
|
CreatedBy uint `gorm:"not null"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||||
|
UpdatedAt time.Time `gorm:"autoUpdateTime"`
|
||||||
|
DeletedAt gorm.DeletedAt `gorm:"index" json:"-"`
|
||||||
|
|
||||||
|
ProjectFlock ProjectFlock `gorm:"foreignKey:ProjectFlocId;references:Id"`
|
||||||
|
CreatedUser User `gorm:"foreignKey:CreatedBy;references:Id"`
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,26 @@
|
|||||||
|
package entities
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
EntityTypeProjectFlockKandang = "PROJECT_FLOCK_KANDANG"
|
||||||
|
)
|
||||||
|
|
||||||
|
type StockAvailability struct {
|
||||||
|
Id uint `gorm:"primaryKey"`
|
||||||
|
EntityType string `gorm:"size:50;not null"`
|
||||||
|
EntityId uint `gorm:"not null"`
|
||||||
|
ProductId uint `gorm:"not null"`
|
||||||
|
Quantity float64 `gorm:"not null;default:0"`
|
||||||
|
ReservedQuantity float64 `gorm:"not null;default:0"`
|
||||||
|
Unit string `gorm:"size:20"`
|
||||||
|
LastUpdated time.Time `gorm:"autoUpdateTime"`
|
||||||
|
CreatedAt time.Time `gorm:"autoCreateTime"`
|
||||||
|
DeletedAt gorm.DeletedAt `gorm:"index" json:"-"`
|
||||||
|
|
||||||
|
Product *Product `gorm:"foreignKey:ProductId;references:Id"`
|
||||||
|
}
|
||||||
@@ -8,6 +8,7 @@ import (
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
LogTypeAdjustment = "ADJUSTMENT"
|
LogTypeAdjustment = "ADJUSTMENT"
|
||||||
|
LogTypeTransfer = "TRANSFER"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import (
|
|||||||
rProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
rProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
||||||
rproduct "gitlab.com/mbugroup/lti-api.git/internal/modules/master/products/repositories"
|
rproduct "gitlab.com/mbugroup/lti-api.git/internal/modules/master/products/repositories"
|
||||||
rWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
rWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
||||||
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/stock-logs/repositories"
|
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
|
|
||||||
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
||||||
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import (
|
|||||||
ProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
ProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
||||||
productRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/products/repositories"
|
productRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/products/repositories"
|
||||||
warehouseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
warehouseRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
||||||
stockLogsRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/stock-logs/repositories"
|
stockLogsRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
|
||||||
|
|||||||
+2
-1
@@ -28,7 +28,6 @@ func (u *ProductWarehouseController) GetAll(c *fiber.Ctx) error {
|
|||||||
Limit: c.QueryInt("limit", 10),
|
Limit: c.QueryInt("limit", 10),
|
||||||
ProductId: uint(c.QueryInt("product_id", 0)),
|
ProductId: uint(c.QueryInt("product_id", 0)),
|
||||||
WarehouseId: uint(c.QueryInt("warehouse_id", 0)),
|
WarehouseId: uint(c.QueryInt("warehouse_id", 0)),
|
||||||
Flag: c.Query("flag"),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
result, totalResults, err := u.ProductWarehouseService.GetAll(c, query)
|
result, totalResults, err := u.ProductWarehouseService.GetAll(c, query)
|
||||||
@@ -72,3 +71,5 @@ func (u *ProductWarehouseController) GetOne(c *fiber.Ctx) error {
|
|||||||
Data: dto.ToProductWarehouseListDTO(*result),
|
Data: dto.ToProductWarehouseListDTO(*result),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
+1
-20
@@ -34,14 +34,7 @@ func NewProductWarehouseService(repo repository.ProductWarehouseRepository, vali
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (s productWarehouseService) withRelations(db *gorm.DB) *gorm.DB {
|
func (s productWarehouseService) withRelations(db *gorm.DB) *gorm.DB {
|
||||||
return db.
|
return db.Preload("Product.Flags").Preload("Product").Preload("Warehouse").Preload("CreatedUser")
|
||||||
Preload("Product").
|
|
||||||
Preload("Product.Flags").
|
|
||||||
Preload("Warehouse").
|
|
||||||
Preload("Warehouse.Location").
|
|
||||||
Preload("Warehouse.Kandang").
|
|
||||||
Preload("Warehouse.Area").
|
|
||||||
Preload("CreatedUser")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s productWarehouseService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.ProductWarehouse, int64, error) {
|
func (s productWarehouseService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.ProductWarehouse, int64, error) {
|
||||||
@@ -62,12 +55,6 @@ func (s productWarehouseService) GetAll(c *fiber.Ctx, params *validation.Query)
|
|||||||
db = db.Where("warehouse_id = ?", params.WarehouseId)
|
db = db.Where("warehouse_id = ?", params.WarehouseId)
|
||||||
}
|
}
|
||||||
|
|
||||||
if params.Flag != "" {
|
|
||||||
db = db.Joins("JOIN products ON products.id = product_warehouses.product_id")
|
|
||||||
db = db.Joins("JOIN flags ON flags.flagable_id = products.id AND flags.flagable_type = ?", "products")
|
|
||||||
db = db.Where("flags.name = ?", params.Flag)
|
|
||||||
}
|
|
||||||
|
|
||||||
return db.Order("created_at DESC").Order("updated_at DESC")
|
return db.Order("created_at DESC").Order("updated_at DESC")
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -75,11 +62,6 @@ func (s productWarehouseService) GetAll(c *fiber.Ctx, params *validation.Query)
|
|||||||
s.Log.Errorf("Failed to get productWarehouses: %+v", err)
|
s.Log.Errorf("Failed to get productWarehouses: %+v", err)
|
||||||
return nil, 0, err
|
return nil, 0, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(productWarehouses) == 0 {
|
|
||||||
return nil, 0, fiber.NewError(fiber.StatusNotFound, "ProductWarehouses not found")
|
|
||||||
}
|
|
||||||
|
|
||||||
return productWarehouses, total, nil
|
return productWarehouses, total, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -88,7 +70,6 @@ func (s productWarehouseService) GetOne(c *fiber.Ctx, id uint) (*entity.ProductW
|
|||||||
if errors.Is(err, gorm.ErrRecordNotFound) {
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
return nil, fiber.NewError(fiber.StatusNotFound, "ProductWarehouse not found")
|
return nil, fiber.NewError(fiber.StatusNotFound, "ProductWarehouse not found")
|
||||||
}
|
}
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
s.Log.Errorf("Failed get productWarehouse by id: %+v", err)
|
s.Log.Errorf("Failed get productWarehouse by id: %+v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
|
|||||||
+4
-5
@@ -13,9 +13,8 @@ type Update struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
type Query struct {
|
type Query struct {
|
||||||
Page int `query:"page" validate:"omitempty,number,min=1"`
|
Page int `query:"page" validate:"omitempty,number,min=1"`
|
||||||
Limit int `query:"limit" validate:"omitempty,number,min=1,max=100"`
|
Limit int `query:"limit" validate:"omitempty,number,min=1,max=100"`
|
||||||
ProductId uint `query:"product_id" validate:"omitempty,number,min=1"`
|
ProductId uint `query:"product_id" validate:"omitempty,number,min=1"`
|
||||||
WarehouseId uint `query:"warehouse_id" validate:"omitempty,number,min=1"`
|
WarehouseId uint `query:"warehouse_id" validate:"omitempty,number,min=1"`
|
||||||
Flag string `query:"flag" validate:"omitempty"`
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -57,17 +57,17 @@ type TransferDetailDTO struct {
|
|||||||
|
|
||||||
// Detail produk
|
// Detail produk
|
||||||
type TransferDetailItemDTO struct {
|
type TransferDetailItemDTO struct {
|
||||||
Id uint64 `json:"id"`
|
Id uint64 `json:"id"`
|
||||||
Product *ProductDTO `json:"product,omitempty"`
|
ProductId uint64 `json:"product_id"`
|
||||||
Quantity float64 `json:"quantity"`
|
Quantity float64 `json:"quantity"`
|
||||||
BeforeQuantity float64 `json:"before_quantity"`
|
BeforeQuantity float64 `json:"before_quantity"`
|
||||||
AfterQuantity float64 `json:"after_quantity"`
|
AfterQuantity float64 `json:"after_quantity"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Delivery ekspedisi
|
// Delivery ekspedisi
|
||||||
type TransferDeliveryDTO struct {
|
type TransferDeliveryDTO struct {
|
||||||
Id uint64 `json:"id"`
|
Id uint64 `json:"id"`
|
||||||
Supplier *SupplierDTO `json:"supplier,omitempty"`
|
SupplierId uint64 `json:"supplier_id"`
|
||||||
VehiclePlate string `json:"vehicle_plate"`
|
VehiclePlate string `json:"vehicle_plate"`
|
||||||
DriverName string `json:"driver_name"`
|
DriverName string `json:"driver_name"`
|
||||||
DocumentNumber string `json:"document_number"`
|
DocumentNumber string `json:"document_number"`
|
||||||
@@ -83,16 +83,6 @@ type TransferDeliveryItemDTO struct {
|
|||||||
Quantity float64 `json:"quantity"`
|
Quantity float64 `json:"quantity"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProductDTO struct {
|
|
||||||
Id uint64 `json:"id"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type SupplierDTO struct {
|
|
||||||
Id uint64 `json:"id"`
|
|
||||||
Name string `json:"name"`
|
|
||||||
}
|
|
||||||
|
|
||||||
// === Mapper Functions ===
|
// === Mapper Functions ===
|
||||||
|
|
||||||
func ToTransferBaseDTO(e entity.StockTransfer) TransferBaseDTO {
|
func ToTransferBaseDTO(e entity.StockTransfer) TransferBaseDTO {
|
||||||
@@ -124,26 +114,6 @@ func toAreaDTO(a *entity.Area) *AreaDTO {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func toProductDTO(p *entity.Product) *ProductDTO {
|
|
||||||
if p == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return &ProductDTO{
|
|
||||||
Id: uint64(p.Id),
|
|
||||||
Name: p.Name,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func toSupplierDTO(s *entity.Supplier) *SupplierDTO {
|
|
||||||
if s == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return &SupplierDTO{
|
|
||||||
Id: uint64(s.Id),
|
|
||||||
Name: s.Name,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func toLocationDTO(l *entity.Location) *LocationDTO {
|
func toLocationDTO(l *entity.Location) *LocationDTO {
|
||||||
if l == nil {
|
if l == nil {
|
||||||
return nil
|
return nil
|
||||||
@@ -172,19 +142,19 @@ func ToTransferListDTO(e entity.StockTransfer) TransferListDTO {
|
|||||||
mapped := userDTO.ToUserBaseDTO(*e.CreatedUser)
|
mapped := userDTO.ToUserBaseDTO(*e.CreatedUser)
|
||||||
createdUser = &mapped
|
createdUser = &mapped
|
||||||
}
|
}
|
||||||
|
// Map details
|
||||||
var details []TransferDetailItemDTO
|
var details []TransferDetailItemDTO
|
||||||
for _, d := range e.Details {
|
for _, d := range e.Details {
|
||||||
details = append(details, TransferDetailItemDTO{
|
details = append(details, TransferDetailItemDTO{
|
||||||
Id: d.Id,
|
Id: d.Id,
|
||||||
Product: toProductDTO(d.Product),
|
ProductId: d.ProductId,
|
||||||
Quantity: d.Quantity,
|
Quantity: d.Quantity,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
// Map deliveries
|
||||||
var deliveries []TransferDeliveryDTO
|
var deliveries []TransferDeliveryDTO
|
||||||
for _, del := range e.Deliveries {
|
for _, del := range e.Deliveries {
|
||||||
|
// Map delivery items
|
||||||
var items []TransferDeliveryItemDTO
|
var items []TransferDeliveryItemDTO
|
||||||
for _, item := range del.Items {
|
for _, item := range del.Items {
|
||||||
items = append(items, TransferDeliveryItemDTO{
|
items = append(items, TransferDeliveryItemDTO{
|
||||||
@@ -195,8 +165,8 @@ func ToTransferListDTO(e entity.StockTransfer) TransferListDTO {
|
|||||||
}
|
}
|
||||||
deliveries = append(deliveries, TransferDeliveryDTO{
|
deliveries = append(deliveries, TransferDeliveryDTO{
|
||||||
Id: del.Id,
|
Id: del.Id,
|
||||||
|
SupplierId: del.SupplierId,
|
||||||
VehiclePlate: del.VehiclePlate,
|
VehiclePlate: del.VehiclePlate,
|
||||||
Supplier: toSupplierDTO(del.Supplier),
|
|
||||||
DriverName: del.DriverName,
|
DriverName: del.DriverName,
|
||||||
DocumentNumber: del.DocumentNumber,
|
DocumentNumber: del.DocumentNumber,
|
||||||
DocumentPath: del.DocumentPath,
|
DocumentPath: del.DocumentPath,
|
||||||
@@ -228,9 +198,9 @@ func ToTransferDetailDTO(e entity.StockTransfer) TransferDetailDTO {
|
|||||||
var details []TransferDetailItemDTO
|
var details []TransferDetailItemDTO
|
||||||
for _, d := range e.Details {
|
for _, d := range e.Details {
|
||||||
details = append(details, TransferDetailItemDTO{
|
details = append(details, TransferDetailItemDTO{
|
||||||
Id: d.Id,
|
Id: d.Id,
|
||||||
Product: toProductDTO(d.Product),
|
ProductId: d.ProductId,
|
||||||
Quantity: d.Quantity,
|
Quantity: d.Quantity,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
// Map deliveries
|
// Map deliveries
|
||||||
@@ -238,7 +208,7 @@ func ToTransferDetailDTO(e entity.StockTransfer) TransferDetailDTO {
|
|||||||
for _, del := range e.Deliveries {
|
for _, del := range e.Deliveries {
|
||||||
deliveries = append(deliveries, TransferDeliveryDTO{
|
deliveries = append(deliveries, TransferDeliveryDTO{
|
||||||
Id: del.Id,
|
Id: del.Id,
|
||||||
Supplier: toSupplierDTO(del.Supplier),
|
SupplierId: del.SupplierId,
|
||||||
VehiclePlate: del.VehiclePlate,
|
VehiclePlate: del.VehiclePlate,
|
||||||
DriverName: del.DriverName,
|
DriverName: del.DriverName,
|
||||||
DocumentNumber: del.DocumentNumber,
|
DocumentNumber: del.DocumentNumber,
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import (
|
|||||||
rStockTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/repositories"
|
rStockTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/repositories"
|
||||||
sTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/services"
|
sTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/services"
|
||||||
rSupplier "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/repositories"
|
rSupplier "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/repositories"
|
||||||
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/stock-logs/repositories"
|
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
||||||
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import (
|
|||||||
rStockTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/repositories"
|
rStockTransfer "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/repositories"
|
||||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/validations"
|
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/transfers/validations"
|
||||||
rSupplier "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/repositories"
|
rSupplier "gitlab.com/mbugroup/lti-api.git/internal/modules/master/suppliers/repositories"
|
||||||
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/stock-logs/repositories"
|
rStockLogs "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
|
|
||||||
"github.com/go-playground/validator/v10"
|
"github.com/go-playground/validator/v10"
|
||||||
@@ -60,8 +60,6 @@ func (s transferService) withRelations(db *gorm.DB) *gorm.DB {
|
|||||||
Preload("ToWarehouse.Location").
|
Preload("ToWarehouse.Location").
|
||||||
Preload("ToWarehouse.Area").
|
Preload("ToWarehouse.Area").
|
||||||
Preload("Details").
|
Preload("Details").
|
||||||
Preload("Details.Product").
|
|
||||||
Preload("Deliveries.Supplier").
|
|
||||||
Preload("Deliveries.Items")
|
Preload("Deliveries.Items")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ type KandangRepository interface {
|
|||||||
PicExists(ctx context.Context, areaId uint) (bool, error)
|
PicExists(ctx context.Context, areaId uint) (bool, error)
|
||||||
NameExists(ctx context.Context, name string, excludeID *uint) (bool, error)
|
NameExists(ctx context.Context, name string, excludeID *uint) (bool, error)
|
||||||
ProjectFlockExists(ctx context.Context, projectFlockID uint) (bool, error)
|
ProjectFlockExists(ctx context.Context, projectFlockID uint) (bool, error)
|
||||||
|
GetFirstByProjectFlockID(ctx context.Context, projectFlockID uint) (*entity.Kandang, error)
|
||||||
HasActiveKandangForProjectFlock(ctx context.Context, projectFlockID uint, excludeID *uint) (bool, error)
|
HasActiveKandangForProjectFlock(ctx context.Context, projectFlockID uint, excludeID *uint) (bool, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -69,3 +70,14 @@ func (r *KandangRepositoryImpl) HasActiveKandangForProjectFlock(ctx context.Cont
|
|||||||
}
|
}
|
||||||
return count > 0, nil
|
return count > 0, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *KandangRepositoryImpl) GetFirstByProjectFlockID(ctx context.Context, projectFlockID uint) (*entity.Kandang, error) {
|
||||||
|
kandang := new(entity.Kandang)
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Where("project_flock_id = ?", projectFlockID).
|
||||||
|
First(kandang).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return kandang, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ type WarehouseRepository interface {
|
|||||||
KandangExists(ctx context.Context, kandangId uint) (bool, error)
|
KandangExists(ctx context.Context, kandangId uint) (bool, error)
|
||||||
NameExists(ctx context.Context, name string, excludeID *uint) (bool, error)
|
NameExists(ctx context.Context, name string, excludeID *uint) (bool, error)
|
||||||
IdExists(ctx context.Context, id uint) (bool, error)
|
IdExists(ctx context.Context, id uint) (bool, error)
|
||||||
|
GetByKandangID(ctx context.Context, kandangId uint) (*entity.Warehouse, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
type WarehouseRepositoryImpl struct {
|
type WarehouseRepositoryImpl struct {
|
||||||
@@ -47,3 +48,15 @@ func (r *WarehouseRepositoryImpl) NameExists(ctx context.Context, name string, e
|
|||||||
func (r *WarehouseRepositoryImpl) IdExists(ctx context.Context, id uint) (bool, error) {
|
func (r *WarehouseRepositoryImpl) IdExists(ctx context.Context, id uint) (bool, error) {
|
||||||
return repository.Exists[entity.Warehouse](ctx, r.db, id)
|
return repository.Exists[entity.Warehouse](ctx, r.db, id)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *WarehouseRepositoryImpl) GetByKandangID(ctx context.Context, kandangId uint) (*entity.Warehouse, error) {
|
||||||
|
var warehouse entity.Warehouse
|
||||||
|
err := r.db.WithContext(ctx).
|
||||||
|
Where("kandang_id = ?", kandangId).
|
||||||
|
Where("deleted_at IS NULL").
|
||||||
|
First(&warehouse).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &warehouse, nil
|
||||||
|
}
|
||||||
|
|||||||
@@ -88,7 +88,7 @@ func (u *ChickinController) CreateOne(c *fiber.Ctx) error {
|
|||||||
Code: fiber.StatusCreated,
|
Code: fiber.StatusCreated,
|
||||||
Status: "success",
|
Status: "success",
|
||||||
Message: "Create chickin successfully",
|
Message: "Create chickin successfully",
|
||||||
Data: dto.ToChickinListDTO(*result),
|
Data: result,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -138,3 +138,24 @@ func (u *ChickinController) DeleteOne(c *fiber.Ctx) error {
|
|||||||
Message: "Delete chickin successfully",
|
Message: "Delete chickin successfully",
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (u *ChickinController) Approve(c *fiber.Ctx) error {
|
||||||
|
param := c.Params("id")
|
||||||
|
|
||||||
|
id, err := strconv.Atoi(param)
|
||||||
|
if err != nil {
|
||||||
|
return fiber.NewError(fiber.StatusBadRequest, "Invalid Id")
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := u.ChickinService.Approve(c, uint(id)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
return c.Status(fiber.StatusOK).
|
||||||
|
JSON(response.Success{
|
||||||
|
Code: fiber.StatusOK,
|
||||||
|
Status: "success",
|
||||||
|
Message: "Approve chickin successfully",
|
||||||
|
Data: nil,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,84 +1,135 @@
|
|||||||
package dto
|
package dto
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
userDTO "gitlab.com/mbugroup/lti-api.git/internal/modules/users/dto"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// === DTO Structs ===
|
// === DTO Structs ===
|
||||||
|
|
||||||
type ChickinBaseDTO struct {
|
type ChickinBaseDTO struct {
|
||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
Name string `json:"name"`
|
ProjectFlocId uint `json:"project_floc_id"`
|
||||||
|
ChickInDate time.Time `json:"chick_in_date"`
|
||||||
|
Quantity float64 `json:"quantity"`
|
||||||
|
Note string `json:"note"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChickinSimpleDTO struct {
|
type ChickinSimpleDTO struct {
|
||||||
Id uint `json:"id"`
|
Id uint `json:"id"`
|
||||||
Name string `json:"name"`
|
ProjectFlocId uint `json:"project_floc_id"`
|
||||||
|
ChickInDate time.Time `json:"chick_in_date"`
|
||||||
|
Quantity float64 `json:"quantity"`
|
||||||
|
Note string `json:"note"`
|
||||||
|
CreatedBy uint `json:"created_by"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type UserBaseDTO struct {
|
||||||
|
Id uint `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChickinListDTO struct {
|
type ChickinListDTO struct {
|
||||||
ChickinBaseDTO
|
ChickinBaseDTO
|
||||||
CreatedUser *userDTO.UserBaseDTO `json:"created_user"`
|
ProjectFlock *ProjectFlockDTO `json:"project_flock"`
|
||||||
CreatedAt time.Time `json:"created_at"`
|
CreatedUser *UserBaseDTO `json:"created_user"`
|
||||||
UpdatedAt time.Time `json:"updated_at"`
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ProjectFlockDTO struct {
|
||||||
|
Id uint `json:"id"`
|
||||||
|
Period int `json:"period"`
|
||||||
|
FlockId uint `json:"flock_id"`
|
||||||
|
FlockName string `json:"flock_name"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Mapper Functions ===
|
||||||
|
|
||||||
|
func ToProjectFlockDTO(e entity.ProjectFlock) ProjectFlockDTO {
|
||||||
|
return ProjectFlockDTO{
|
||||||
|
Id: e.Id,
|
||||||
|
Period: e.Period,
|
||||||
|
FlockId: e.FlockId,
|
||||||
|
FlockName: e.Flock.Name,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type ChickinDetailDTO struct {
|
type ChickinDetailDTO struct {
|
||||||
ChickinListDTO
|
ChickinListDTO
|
||||||
}
|
}
|
||||||
|
|
||||||
// === Mapper Functions ===
|
// === Mapper Functions ===
|
||||||
|
|
||||||
func ToChickinBaseDTO(e entity.ProjectChickin) ChickinBaseDTO {
|
func ToChickinBaseDTO(e entity.ProjectChickin) ChickinBaseDTO {
|
||||||
return ChickinBaseDTO{
|
return ChickinBaseDTO{
|
||||||
Id: e.Id,
|
Id: e.Id,
|
||||||
|
ProjectFlocId: e.ProjectFlocId,
|
||||||
}
|
ChickInDate: e.ChickInDate,
|
||||||
|
Quantity: e.Quantity,
|
||||||
|
Note: e.Note,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToChickinSimpleDTO(e entity.ProjectChickin) ChickinSimpleDTO {
|
func ToChickinSimpleDTO(e entity.ProjectChickin) ChickinSimpleDTO {
|
||||||
return ChickinSimpleDTO{
|
return ChickinSimpleDTO{
|
||||||
Id: e.Id,
|
Id: e.Id,
|
||||||
|
ProjectFlocId: e.ProjectFlocId,
|
||||||
}
|
ChickInDate: e.ChickInDate,
|
||||||
|
Quantity: e.Quantity,
|
||||||
|
Note: e.Note,
|
||||||
|
CreatedBy: e.CreatedBy,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToChickinListDTO(e entity.ProjectChickin) ChickinListDTO {
|
func ToChickinListDTO(e entity.ProjectChickin) ChickinListDTO {
|
||||||
var createdUser *userDTO.UserBaseDTO
|
var createdUser *UserBaseDTO
|
||||||
if e.CreatedUser.Id != 0 {
|
if e.CreatedUser.Id != 0 {
|
||||||
mapped := userDTO.ToUserBaseDTO(e.CreatedUser)
|
mapped := ToUserBaseDTO(e.CreatedUser)
|
||||||
createdUser = &mapped
|
createdUser = &mapped
|
||||||
}
|
}
|
||||||
|
|
||||||
return ChickinListDTO{
|
var projectFlock *ProjectFlockDTO
|
||||||
ChickinBaseDTO: ToChickinBaseDTO(e),
|
if e.ProjectFlock.Id != 0 {
|
||||||
CreatedAt: e.CreatedAt,
|
mapped := ToProjectFlockDTO(e.ProjectFlock)
|
||||||
UpdatedAt: e.UpdatedAt,
|
projectFlock = &mapped
|
||||||
CreatedUser: createdUser,
|
}
|
||||||
}
|
|
||||||
|
return ChickinListDTO{
|
||||||
|
ChickinBaseDTO: ToChickinBaseDTO(e),
|
||||||
|
ProjectFlock: projectFlock,
|
||||||
|
CreatedAt: e.CreatedAt,
|
||||||
|
UpdatedAt: e.UpdatedAt,
|
||||||
|
CreatedUser: createdUser,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToChickinListDTOs(e []entity.ProjectChickin) []ChickinListDTO {
|
func ToChickinListDTOs(e []entity.ProjectChickin) []ChickinListDTO {
|
||||||
result := make([]ChickinListDTO, len(e))
|
result := make([]ChickinListDTO, len(e))
|
||||||
for i, r := range e {
|
for i, r := range e {
|
||||||
result[i] = ToChickinListDTO(r)
|
result[i] = ToChickinListDTO(r)
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToChickinSimpleDTOs(e []entity.ProjectChickin) []ChickinSimpleDTO {
|
func ToChickinSimpleDTOs(e []entity.ProjectChickin) []ChickinSimpleDTO {
|
||||||
result := make([]ChickinSimpleDTO, len(e))
|
result := make([]ChickinSimpleDTO, len(e))
|
||||||
for i, r := range e {
|
for i, r := range e {
|
||||||
result[i] = ToChickinSimpleDTO(r)
|
result[i] = ToChickinSimpleDTO(r)
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
func ToChickinDetailDTO(e entity.ProjectChickin) ChickinDetailDTO {
|
func ToChickinDetailDTO(e entity.ProjectChickin) ChickinDetailDTO {
|
||||||
return ChickinDetailDTO{
|
return ChickinDetailDTO{
|
||||||
ChickinListDTO: ToChickinListDTO(e),
|
ChickinListDTO: ToChickinListDTO(e),
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ToUserBaseDTO(e entity.User) UserBaseDTO {
|
||||||
|
return UserBaseDTO{
|
||||||
|
Id: e.Id,
|
||||||
|
Name: e.Name,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -5,8 +5,14 @@ import (
|
|||||||
"github.com/gofiber/fiber/v2"
|
"github.com/gofiber/fiber/v2"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
|
||||||
|
rProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
||||||
|
rKandang "gitlab.com/mbugroup/lti-api.git/internal/modules/master/kandangs/repositories"
|
||||||
|
rWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
||||||
rChickin "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
rChickin "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
||||||
sChickin "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/services"
|
sChickin "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/services"
|
||||||
|
rAuditLog "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
|
|
||||||
|
rProjectFlock "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/repositories"
|
||||||
|
|
||||||
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
rUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/repositories"
|
||||||
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
sUser "gitlab.com/mbugroup/lti-api.git/internal/modules/users/services"
|
||||||
@@ -16,11 +22,16 @@ type ChickinModule struct{}
|
|||||||
|
|
||||||
func (ChickinModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
func (ChickinModule) RegisterRoutes(router fiber.Router, db *gorm.DB, validate *validator.Validate) {
|
||||||
chickinRepo := rChickin.NewChickinRepository(db)
|
chickinRepo := rChickin.NewChickinRepository(db)
|
||||||
|
kandangRepo := rKandang.NewKandangRepository(db)
|
||||||
|
auditlogrepo := rAuditLog.NewAuditLogRepository(db)
|
||||||
|
warehouseRepo := rWarehouse.NewWarehouseRepository(db)
|
||||||
|
projectFlockRepo := rProjectFlock.NewProjectflockRepository(db)
|
||||||
|
productWarehouseRepo := rProductWarehouse.NewProductWarehouseRepository(db)
|
||||||
|
|
||||||
userRepo := rUser.NewUserRepository(db)
|
userRepo := rUser.NewUserRepository(db)
|
||||||
|
|
||||||
chickinService := sChickin.NewChickinService(chickinRepo, validate)
|
chickinService := sChickin.NewChickinService(chickinRepo, kandangRepo, warehouseRepo, productWarehouseRepo, projectFlockRepo, auditlogrepo, validate)
|
||||||
userService := sUser.NewUserService(userRepo, validate)
|
userService := sUser.NewUserService(userRepo, validate)
|
||||||
|
|
||||||
ChickinRoutes(router, userService, chickinService)
|
ChickinRoutes(router, userService, chickinService)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,21 +0,0 @@
|
|||||||
package repository
|
|
||||||
|
|
||||||
import (
|
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
|
||||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
|
||||||
"gorm.io/gorm"
|
|
||||||
)
|
|
||||||
|
|
||||||
type ChickinRepository interface {
|
|
||||||
repository.BaseRepository[entity.ProjectChickin]
|
|
||||||
}
|
|
||||||
|
|
||||||
type ChickinRepositoryImpl struct {
|
|
||||||
*repository.BaseRepositoryImpl[entity.ProjectChickin]
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewChickinRepository(db *gorm.DB) ChickinRepository {
|
|
||||||
return &ChickinRepositoryImpl{
|
|
||||||
BaseRepositoryImpl: repository.NewBaseRepository[entity.ProjectChickin](db),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type ProjectChickinRepository interface {
|
||||||
|
repository.BaseRepository[entity.ProjectChickin]
|
||||||
|
GetFirstByProjectFlockID(ctx context.Context, projectFlockID uint) (*entity.ProjectChickin, error)
|
||||||
|
}
|
||||||
|
|
||||||
|
type ChickinRepositoryImpl struct {
|
||||||
|
*repository.BaseRepositoryImpl[entity.ProjectChickin]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewChickinRepository(db *gorm.DB) ProjectChickinRepository {
|
||||||
|
return &ChickinRepositoryImpl{
|
||||||
|
BaseRepositoryImpl: repository.NewBaseRepository[entity.ProjectChickin](db),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *ChickinRepositoryImpl) GetFirstByProjectFlockID(ctx context.Context, projectFlockID uint) (*entity.ProjectChickin, error) {
|
||||||
|
var chickin entity.ProjectChickin
|
||||||
|
err := r.DB().WithContext(ctx).
|
||||||
|
Where("project_floc_id = ?", projectFlockID).
|
||||||
|
Where("deleted_at IS NULL").
|
||||||
|
First(&chickin).Error
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &chickin, nil
|
||||||
|
}
|
||||||
@@ -25,4 +25,5 @@ func ChickinRoutes(v1 fiber.Router, u user.UserService, s chickin.ChickinService
|
|||||||
route.Get("/:id", ctrl.GetOne)
|
route.Get("/:id", ctrl.GetOne)
|
||||||
route.Patch("/:id", ctrl.UpdateOne)
|
route.Patch("/:id", ctrl.UpdateOne)
|
||||||
route.Delete("/:id", ctrl.DeleteOne)
|
route.Delete("/:id", ctrl.DeleteOne)
|
||||||
|
route.Post("/:id/approve", ctrl.Approve)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,8 +4,14 @@ import (
|
|||||||
"errors"
|
"errors"
|
||||||
|
|
||||||
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
rProductWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/inventory/product-warehouses/repositories"
|
||||||
|
KandangRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/master/kandangs/repositories"
|
||||||
|
rWarehouse "gitlab.com/mbugroup/lti-api.git/internal/modules/master/warehouses/repositories"
|
||||||
repository "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
repository "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/repositories"
|
||||||
|
|
||||||
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/validations"
|
validation "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins/validations"
|
||||||
|
rProjectFlock "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks/repositories"
|
||||||
|
AuditLogRepo "gitlab.com/mbugroup/lti-api.git/internal/modules/shared/repositories"
|
||||||
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
"gitlab.com/mbugroup/lti-api.git/internal/utils"
|
||||||
|
|
||||||
"github.com/go-playground/validator/v10"
|
"github.com/go-playground/validator/v10"
|
||||||
@@ -20,24 +26,39 @@ type ChickinService interface {
|
|||||||
CreateOne(ctx *fiber.Ctx, req *validation.Create) (*entity.ProjectChickin, error)
|
CreateOne(ctx *fiber.Ctx, req *validation.Create) (*entity.ProjectChickin, error)
|
||||||
UpdateOne(ctx *fiber.Ctx, req *validation.Update, id uint) (*entity.ProjectChickin, error)
|
UpdateOne(ctx *fiber.Ctx, req *validation.Update, id uint) (*entity.ProjectChickin, error)
|
||||||
DeleteOne(ctx *fiber.Ctx, id uint) error
|
DeleteOne(ctx *fiber.Ctx, id uint) error
|
||||||
|
Approve(ctx *fiber.Ctx, id uint) error
|
||||||
}
|
}
|
||||||
|
|
||||||
type chickinService struct {
|
type chickinService struct {
|
||||||
Log *logrus.Logger
|
Log *logrus.Logger
|
||||||
Validate *validator.Validate
|
Validate *validator.Validate
|
||||||
Repository repository.ChickinRepository
|
Repository repository.ProjectChickinRepository
|
||||||
|
KandangRepo KandangRepo.KandangRepository
|
||||||
|
WarehouseRepo rWarehouse.WarehouseRepository
|
||||||
|
ProductWarehouseRepo rProductWarehouse.ProductWarehouseRepository
|
||||||
|
ProjectFlockRepo rProjectFlock.ProjectflockRepository
|
||||||
|
AuditLogRepo AuditLogRepo.AuditLogRepository
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewChickinService(repo repository.ChickinRepository, validate *validator.Validate) ChickinService {
|
func NewChickinService(repo repository.ProjectChickinRepository, kandangRepo KandangRepo.KandangRepository, warehouseRepo rWarehouse.WarehouseRepository, productWarehouseRepo rProductWarehouse.ProductWarehouseRepository, projectFlockRepo rProjectFlock.ProjectflockRepository, auditLogRepo AuditLogRepo.AuditLogRepository, validate *validator.Validate) ChickinService {
|
||||||
return &chickinService{
|
return &chickinService{
|
||||||
Log: utils.Log,
|
Log: utils.Log,
|
||||||
Validate: validate,
|
Validate: validate,
|
||||||
Repository: repo,
|
Repository: repo,
|
||||||
|
KandangRepo: kandangRepo,
|
||||||
|
WarehouseRepo: warehouseRepo,
|
||||||
|
ProductWarehouseRepo: productWarehouseRepo,
|
||||||
|
ProjectFlockRepo: projectFlockRepo,
|
||||||
|
AuditLogRepo: auditLogRepo,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s chickinService) withRelations(db *gorm.DB) *gorm.DB {
|
func (s chickinService) withRelations(db *gorm.DB) *gorm.DB {
|
||||||
return db.Preload("CreatedUser")
|
return db.
|
||||||
|
Preload("CreatedUser").
|
||||||
|
Preload("ProjectFlock").
|
||||||
|
Preload("ProjectFlock.ProductCategory")
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s chickinService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.ProjectChickin, int64, error) {
|
func (s chickinService) GetAll(c *fiber.Ctx, params *validation.Query) ([]entity.ProjectChickin, int64, error) {
|
||||||
@@ -79,16 +100,125 @@ func (s *chickinService) CreateOne(c *fiber.Ctx, req *validation.Create) (*entit
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
createBody := &entity.ProjectChickin{
|
// ambil salah satu kandang dari project_floc_id dari kandang repository
|
||||||
ProjectFlocId: 1,
|
kandang, err := s.KandangRepo.GetFirstByProjectFlockID(c.Context(), 1)
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to get kandang: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// ambil warehouse dari kandangid
|
||||||
|
warehouse, err := s.WarehouseRepo.GetByKandangID(c.Context(), kandang.Id)
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to get warehouse: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// getprojectflock id with relation
|
||||||
|
projectFlock, err := s.ProjectFlockRepo.GetByID(
|
||||||
|
c.Context(),
|
||||||
|
req.ProjectFlockId,
|
||||||
|
func(db *gorm.DB) *gorm.DB {
|
||||||
|
return db.Preload("ProductCategory")
|
||||||
|
},
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to get project flock: %+v", err)
|
||||||
|
return nil, fiber.NewError(fiber.StatusNotFound, "Project Flock not found")
|
||||||
|
}
|
||||||
|
// ambil quantity
|
||||||
|
var productWarehouse entity.ProductWarehouse
|
||||||
|
err = s.ProductWarehouseRepo.DB().WithContext(c.Context()).
|
||||||
|
Joins("JOIN products ON products.id = product_warehouses.product_id").
|
||||||
|
Joins("JOIN product_categories ON product_categories.id = products.product_category_id").
|
||||||
|
Where("product_categories.code = ? AND product_warehouses.warehouse_id = ?", projectFlock.ProductCategory.Code, warehouse.Id).
|
||||||
|
Order("created_at DESC").
|
||||||
|
First(&productWarehouse).Error
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
return nil, fiber.NewError(fiber.StatusNotFound, "Product Warehouse not found for the given Project Flock and Warehouse")
|
||||||
|
}
|
||||||
|
s.Log.Errorf("Failed to get product warehouse: %+v", err)
|
||||||
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := s.Repository.CreateOne(c.Context(), createBody, nil); err != nil {
|
if productWarehouse.Quantity < 1 {
|
||||||
|
return nil, fiber.NewError(fiber.StatusBadRequest, "Insufficient product quantity in warehouse")
|
||||||
|
}
|
||||||
|
|
||||||
|
// masukan ke chic in
|
||||||
|
chickinDate, err := utils.ParseDateString(req.ChickInDate)
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to parse chickin date: %+v", err)
|
||||||
|
return nil, fiber.NewError(fiber.StatusBadRequest, "Invalid ChickInDate format")
|
||||||
|
}
|
||||||
|
|
||||||
|
newChickin := &entity.ProjectChickin{
|
||||||
|
ProjectFlocId: req.ProjectFlockId,
|
||||||
|
ChickInDate: chickinDate,
|
||||||
|
Quantity: productWarehouse.Quantity,
|
||||||
|
Note: "",
|
||||||
|
CreatedBy: 1, //todo: ganti dengan
|
||||||
|
}
|
||||||
|
|
||||||
|
err = s.Repository.CreateOne(c.Context(), newChickin, nil)
|
||||||
|
if err != nil {
|
||||||
s.Log.Errorf("Failed to create chickin: %+v", err)
|
s.Log.Errorf("Failed to create chickin: %+v", err)
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
return s.GetOne(c, createBody.Id)
|
// Kurangi quantity di product warehouse
|
||||||
|
updatedQuantity := productWarehouse.Quantity - newChickin.Quantity
|
||||||
|
if updatedQuantity < 0 {
|
||||||
|
updatedQuantity = 0
|
||||||
|
}
|
||||||
|
err = s.ProductWarehouseRepo.PatchOne(c.Context(), productWarehouse.Id, map[string]any{
|
||||||
|
"quantity": updatedQuantity,
|
||||||
|
}, nil)
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to update product warehouse quantity: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// masukan check apakah stock availability ada, jika ada update, jika tidak buat baru
|
||||||
|
stockAvailability := &entity.StockAvailability{
|
||||||
|
EntityType: entity.EntityTypeProjectFlockKandang,
|
||||||
|
ReservedQuantity: productWarehouse.Quantity,
|
||||||
|
EntityId: req.ProjectFlockId, //todo: nanti pakek projct flock kandang id
|
||||||
|
ProductId: productWarehouse.ProductId,
|
||||||
|
}
|
||||||
|
|
||||||
|
var existingStockAvailability entity.StockAvailability
|
||||||
|
err = s.ProductWarehouseRepo.DB().WithContext(c.Context()).
|
||||||
|
Where("entity_type = ? AND entity_id = ? AND product_id = ?", stockAvailability.EntityType, stockAvailability.EntityId, stockAvailability.ProductId).
|
||||||
|
First(&existingStockAvailability).Error
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
// buat baru
|
||||||
|
stockAvailability.ReservedQuantity = newChickin.Quantity
|
||||||
|
stockAvailability.Quantity = 0
|
||||||
|
err = s.ProductWarehouseRepo.DB().WithContext(c.Context()).Create(stockAvailability).Error
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to create stock availability: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
s.Log.Errorf("Failed to get stock availability: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// update existing
|
||||||
|
newQuantity := existingStockAvailability.ReservedQuantity + newChickin.Quantity
|
||||||
|
err = s.ProductWarehouseRepo.DB().WithContext(c.Context()).
|
||||||
|
Model(&existingStockAvailability).
|
||||||
|
Update("reserved_quantity", newQuantity).Error
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to update stock availability: %+v", err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s chickinService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint) (*entity.ProjectChickin, error) {
|
func (s chickinService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint) (*entity.ProjectChickin, error) {
|
||||||
@@ -98,10 +228,9 @@ func (s chickinService) UpdateOne(c *fiber.Ctx, req *validation.Update, id uint)
|
|||||||
|
|
||||||
updateBody := make(map[string]any)
|
updateBody := make(map[string]any)
|
||||||
|
|
||||||
if req.Name != nil {
|
if req.ChickInDate != "" {
|
||||||
updateBody["name"] = *req.Name
|
updateBody["chick_in_date"] = req.ChickInDate
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(updateBody) == 0 {
|
if len(updateBody) == 0 {
|
||||||
return s.GetOne(c, id)
|
return s.GetOne(c, id)
|
||||||
}
|
}
|
||||||
@@ -125,5 +254,41 @@ func (s chickinService) DeleteOne(c *fiber.Ctx, id uint) error {
|
|||||||
s.Log.Errorf("Failed to delete chickin: %+v", err)
|
s.Log.Errorf("Failed to delete chickin: %+v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *chickinService) Approve(c *fiber.Ctx, id uint) error {
|
||||||
|
|
||||||
|
chickin, err := s.Repository.GetByID(
|
||||||
|
c.Context(),
|
||||||
|
id,
|
||||||
|
nil,
|
||||||
|
)
|
||||||
|
if errors.Is(err, gorm.ErrRecordNotFound) {
|
||||||
|
return fiber.NewError(fiber.StatusNotFound, "Chickin not found")
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed get chickin by id: %+v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
//pindahkan stock dari reserved ke actual stock
|
||||||
|
// get stock avaibility untuk di update
|
||||||
|
var stockAvailability entity.StockAvailability
|
||||||
|
err = s.ProductWarehouseRepo.DB().WithContext(c.Context()).
|
||||||
|
Where("entity_type = ? AND entity_id = ? ", entity.EntityTypeProjectFlockKandang, chickin.ProjectFlocId).
|
||||||
|
First(&stockAvailability).Error
|
||||||
|
if err != nil {
|
||||||
|
s.Log.Errorf("Failed to get stock availability: %+v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
newReservedQuantity := stockAvailability.ReservedQuantity - chickin.Quantity
|
||||||
|
if newReservedQuantity < 0 {
|
||||||
|
newReservedQuantity = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
package validation
|
package validation
|
||||||
|
|
||||||
type Create struct {
|
type Create struct {
|
||||||
Name string `json:"name" validate:"required_strict,min=3"`
|
ProjectFlockId uint `json:"project_flock_id" validate:"required,number,min=1"`
|
||||||
|
ChickInDate string `json:"chick_in_date" validate:"required,datetime=2006-01-02"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Update struct {
|
type Update struct {
|
||||||
Name *string `json:"name,omitempty" validate:"omitempty"`
|
ChickInDate string `json:"chick_in_date" validate:"required,datetime=2006-01-02"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type Query struct {
|
type Query struct {
|
||||||
|
|||||||
@@ -7,9 +7,9 @@ import (
|
|||||||
"github.com/gofiber/fiber/v2"
|
"github.com/gofiber/fiber/v2"
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
|
||||||
|
chickins "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins"
|
||||||
projectflocks "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks"
|
projectflocks "gitlab.com/mbugroup/lti-api.git/internal/modules/production/project_flocks"
|
||||||
recordings "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings"
|
recordings "gitlab.com/mbugroup/lti-api.git/internal/modules/production/recordings"
|
||||||
chickins "gitlab.com/mbugroup/lti-api.git/internal/modules/production/chickins"
|
|
||||||
// MODULE IMPORTS
|
// MODULE IMPORTS
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AuditLogRepository interface {
|
||||||
|
repository.BaseRepository[entity.AuditLog]
|
||||||
|
}
|
||||||
|
|
||||||
|
type AuditLogRepositoryImpl struct {
|
||||||
|
*repository.BaseRepositoryImpl[entity.AuditLog]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewAuditLogRepository(db *gorm.DB) AuditLogRepository {
|
||||||
|
return &AuditLogRepositoryImpl{
|
||||||
|
BaseRepositoryImpl: repository.NewBaseRepository[entity.AuditLog](db),
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
package repository
|
||||||
|
|
||||||
|
import (
|
||||||
|
"gitlab.com/mbugroup/lti-api.git/internal/common/repository"
|
||||||
|
entity "gitlab.com/mbugroup/lti-api.git/internal/entities"
|
||||||
|
"gorm.io/gorm"
|
||||||
|
)
|
||||||
|
|
||||||
|
type StockAvailabilityRepository interface {
|
||||||
|
repository.BaseRepository[entity.StockAvailability]
|
||||||
|
}
|
||||||
|
|
||||||
|
type StockAvailabilityRepositoryImpl struct {
|
||||||
|
*repository.BaseRepositoryImpl[entity.StockAvailability]
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewStockAvailabilityRepository(db *gorm.DB) StockAvailabilityRepository {
|
||||||
|
return &StockAvailabilityRepositoryImpl{
|
||||||
|
BaseRepositoryImpl: repository.NewBaseRepository[entity.StockAvailability](db),
|
||||||
|
}
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user