Files
pig-farm-controller/internal/app/service/pig_farm_service.go

315 lines
9.0 KiB
Go
Raw Normal View History

package service
import (
2025-10-03 22:33:43 +08:00
"errors"
2025-10-04 01:28:05 +08:00
"fmt"
2025-10-03 22:33:43 +08:00
"git.huangwc.com/pig/pig-farm-controller/internal/app/dto"
domain_pig "git.huangwc.com/pig/pig-farm-controller/internal/domain/pig"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/logs"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/models"
"git.huangwc.com/pig/pig-farm-controller/internal/infra/repository"
2025-10-03 22:33:43 +08:00
"gorm.io/gorm"
)
// PigFarmService 提供了猪场资产管理的业务逻辑
type PigFarmService interface {
// PigHouse methods
CreatePigHouse(name, description string) (*models.PigHouse, error)
GetPigHouseByID(id uint) (*models.PigHouse, error)
ListPigHouses() ([]models.PigHouse, error)
UpdatePigHouse(id uint, name, description string) (*models.PigHouse, error)
DeletePigHouse(id uint) error
// Pen methods
2025-10-03 23:58:37 +08:00
CreatePen(penNumber string, houseID uint, capacity int) (*models.Pen, error)
GetPenByID(id uint) (*dto.PenResponse, error)
ListPens() ([]*dto.PenResponse, error)
UpdatePen(id uint, penNumber string, houseID uint, capacity int, status models.PenStatus) (*models.Pen, error)
DeletePen(id uint) error
2025-10-04 01:28:05 +08:00
// UpdatePenStatus 更新猪栏状态
UpdatePenStatus(id uint, newStatus models.PenStatus) (*models.Pen, error)
}
type pigFarmService struct {
2025-10-05 17:46:03 +08:00
logger *logs.Logger
farmRepository repository.PigFarmRepository
penRepository repository.PigPenRepository
batchRepository repository.PigBatchRepository
pigBatchService domain_pig.PigBatchService // Add domain PigBatchService dependency
uow repository.UnitOfWork // 工作单元,用于事务管理
}
// NewPigFarmService 创建一个新的 PigFarmService 实例
2025-10-05 17:46:03 +08:00
func NewPigFarmService(farmRepository repository.PigFarmRepository,
penRepository repository.PigPenRepository,
batchRepository repository.PigBatchRepository,
pigBatchService domain_pig.PigBatchService,
2025-10-05 17:46:03 +08:00
uow repository.UnitOfWork,
logger *logs.Logger) PigFarmService {
return &pigFarmService{
2025-10-05 17:46:03 +08:00
logger: logger,
farmRepository: farmRepository,
penRepository: penRepository,
batchRepository: batchRepository,
pigBatchService: pigBatchService,
2025-10-05 17:46:03 +08:00
uow: uow,
}
}
// --- PigHouse Implementation ---
func (s *pigFarmService) CreatePigHouse(name, description string) (*models.PigHouse, error) {
house := &models.PigHouse{
Name: name,
Description: description,
}
2025-10-05 17:42:27 +08:00
err := s.farmRepository.CreatePigHouse(house)
return house, err
}
func (s *pigFarmService) GetPigHouseByID(id uint) (*models.PigHouse, error) {
2025-10-05 17:42:27 +08:00
return s.farmRepository.GetPigHouseByID(id)
}
func (s *pigFarmService) ListPigHouses() ([]models.PigHouse, error) {
2025-10-05 17:42:27 +08:00
return s.farmRepository.ListPigHouses()
}
func (s *pigFarmService) UpdatePigHouse(id uint, name, description string) (*models.PigHouse, error) {
house := &models.PigHouse{
Model: gorm.Model{ID: id},
Name: name,
Description: description,
}
2025-10-05 17:42:27 +08:00
rowsAffected, err := s.farmRepository.UpdatePigHouse(house)
if err != nil {
return nil, err
}
2025-10-04 00:58:29 +08:00
if rowsAffected == 0 {
return nil, ErrHouseNotFound
}
// 返回更新后的完整信息
2025-10-05 17:42:27 +08:00
return s.farmRepository.GetPigHouseByID(id)
}
func (s *pigFarmService) DeletePigHouse(id uint) error {
2025-10-03 22:33:43 +08:00
// 业务逻辑:检查猪舍是否包含猪栏
2025-10-05 17:42:27 +08:00
penCount, err := s.farmRepository.CountPensInHouse(id)
2025-10-03 22:33:43 +08:00
if err != nil {
return err
}
if penCount > 0 {
return ErrHouseContainsPens
}
// 调用仓库层进行删除
2025-10-05 17:42:27 +08:00
rowsAffected, err := s.farmRepository.DeletePigHouse(id)
2025-10-04 00:58:29 +08:00
if err != nil {
return err
}
if rowsAffected == 0 {
return ErrHouseNotFound
2025-10-03 22:33:43 +08:00
}
2025-10-04 00:58:29 +08:00
return nil
}
// --- Pen Implementation ---
2025-10-03 23:58:37 +08:00
func (s *pigFarmService) CreatePen(penNumber string, houseID uint, capacity int) (*models.Pen, error) {
2025-10-03 22:33:43 +08:00
// 业务逻辑:验证所属猪舍是否存在
2025-10-05 17:42:27 +08:00
_, err := s.farmRepository.GetPigHouseByID(houseID)
2025-10-03 22:33:43 +08:00
if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, ErrHouseNotFound
}
return nil, err
}
pen := &models.Pen{
PenNumber: penNumber,
HouseID: houseID,
Capacity: capacity,
2025-10-03 23:58:37 +08:00
Status: models.PenStatusEmpty,
}
2025-10-05 17:42:27 +08:00
err = s.penRepository.CreatePen(pen)
return pen, err
}
func (s *pigFarmService) GetPenByID(id uint) (*dto.PenResponse, error) {
pen, err := s.penRepository.GetPenByID(id)
if err != nil {
return nil, err
}
currentPigCount, err := s.pigBatchService.GetCurrentPigsInPen(id)
if err != nil {
s.logger.Errorf("获取猪栏 %d 存栏量失败: %v", id, err)
currentPigCount = 0 // 如果获取计数时出错则默认为0
}
response := &dto.PenResponse{
ID: pen.ID,
PenNumber: pen.PenNumber,
HouseID: pen.HouseID,
Capacity: pen.Capacity,
Status: pen.Status,
CurrentPigCount: currentPigCount,
}
if pen.PigBatchID != nil {
response.PigBatchID = pen.PigBatchID
}
return response, nil
}
func (s *pigFarmService) ListPens() ([]*dto.PenResponse, error) {
pens, err := s.penRepository.ListPens()
if err != nil {
return nil, err
}
var response []*dto.PenResponse
for _, pen := range pens {
currentPigCount, err := s.pigBatchService.GetCurrentPigsInPen(pen.ID)
if err != nil {
s.logger.Errorf("获取猪栏 %d 存栏量失败: %v", pen.ID, err)
currentPigCount = 0 // 如果获取计数时出错则默认为0
}
penResponse := &dto.PenResponse{
ID: pen.ID,
PenNumber: pen.PenNumber,
HouseID: pen.HouseID,
Capacity: pen.Capacity,
Status: pen.Status,
CurrentPigCount: currentPigCount,
}
if pen.PigBatchID != nil {
penResponse.PigBatchID = pen.PigBatchID
}
response = append(response, penResponse)
}
return response, nil
}
func (s *pigFarmService) UpdatePen(id uint, penNumber string, houseID uint, capacity int, status models.PenStatus) (*models.Pen, error) {
2025-10-03 22:33:43 +08:00
// 业务逻辑:验证所属猪舍是否存在
2025-10-05 17:42:27 +08:00
_, err := s.farmRepository.GetPigHouseByID(houseID)
2025-10-03 22:33:43 +08:00
if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return nil, ErrHouseNotFound
}
return nil, err
}
pen := &models.Pen{
Model: gorm.Model{ID: id},
PenNumber: penNumber,
HouseID: houseID,
Capacity: capacity,
Status: status,
}
2025-10-05 17:42:27 +08:00
rowsAffected, err := s.penRepository.UpdatePen(pen)
if err != nil {
return nil, err
}
2025-10-04 00:58:29 +08:00
if rowsAffected == 0 {
return nil, ErrPenNotFound
}
// 返回更新后的完整信息
2025-10-05 17:42:27 +08:00
return s.penRepository.GetPenByID(id)
}
func (s *pigFarmService) DeletePen(id uint) error {
2025-10-03 22:33:43 +08:00
// 业务逻辑:检查猪栏是否被活跃批次使用
2025-10-05 17:42:27 +08:00
pen, err := s.penRepository.GetPenByID(id)
2025-10-03 22:33:43 +08:00
if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
2025-10-04 00:58:29 +08:00
return ErrPenNotFound // 猪栏不存在
2025-10-03 22:33:43 +08:00
}
return err
}
// 检查猪栏是否关联了活跃批次
2025-10-04 00:58:29 +08:00
// 注意pen.PigBatchID 是指针类型,需要检查是否为 nil
if pen.PigBatchID != nil && *pen.PigBatchID != 0 {
2025-10-05 17:46:03 +08:00
pigBatch, err := s.batchRepository.GetPigBatchByID(*pen.PigBatchID)
2025-10-03 22:33:43 +08:00
if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) {
return err
}
// 如果批次活跃,则不能删除猪栏
2025-10-04 00:58:29 +08:00
if pigBatch != nil && pigBatch.IsActive() {
2025-10-03 22:33:43 +08:00
return ErrPenInUse
}
}
// 调用仓库层进行删除
2025-10-05 17:42:27 +08:00
rowsAffected, err := s.penRepository.DeletePen(id)
2025-10-04 00:58:29 +08:00
if err != nil {
return err
}
if rowsAffected == 0 {
return ErrPenNotFound
2025-10-03 22:33:43 +08:00
}
2025-10-04 00:58:29 +08:00
return nil
}
2025-10-04 01:28:05 +08:00
// UpdatePenStatus 更新猪栏状态
func (s *pigFarmService) UpdatePenStatus(id uint, newStatus models.PenStatus) (*models.Pen, error) {
var updatedPen *models.Pen
err := s.uow.ExecuteInTransaction(func(tx *gorm.DB) error {
2025-10-05 17:42:27 +08:00
pen, err := s.penRepository.GetPenByIDTx(tx, id)
2025-10-04 01:28:05 +08:00
if err != nil {
if errors.Is(err, gorm.ErrRecordNotFound) {
return ErrPenNotFound
}
s.logger.Errorf("更新猪栏状态失败: 获取猪栏 %d 信息错误: %v", id, err)
return fmt.Errorf("获取猪栏 %d 信息失败: %w", id, err)
}
// 业务逻辑:根据猪栏的 PigBatchID 和当前状态,判断是否允许设置为 newStatus
2025-10-04 01:31:35 +08:00
if pen.PigBatchID != nil && *pen.PigBatchID != 0 { // 猪栏已被批次使用
if newStatus == models.PenStatusEmpty { // 猪栏已被批次使用,不能直接设置为空闲
2025-10-04 01:28:05 +08:00
return ErrPenStatusInvalidForOccupiedPen
}
2025-10-04 01:31:35 +08:00
} else { // 猪栏未被批次使用 (PigBatchID == nil)
if newStatus == models.PenStatusOccupied { // 猪栏未被批次使用,不能设置为使用中
2025-10-04 01:28:05 +08:00
return ErrPenStatusInvalidForUnoccupiedPen
}
}
// 如果新状态与旧状态相同,则无需更新
if pen.Status == newStatus {
updatedPen = pen // 返回原始猪栏,因为没有实际更新
return nil
}
updates := map[string]interface{}{
"status": newStatus,
}
2025-10-05 18:00:50 +08:00
if err := s.penRepository.UpdatePenFieldsTx(tx, id, updates); err != nil {
2025-10-04 01:28:05 +08:00
s.logger.Errorf("更新猪栏 %d 状态失败: %v", id, err)
return fmt.Errorf("更新猪栏 %d 状态失败: %w", id, err)
}
// 获取更新后的猪栏信息
2025-10-05 17:42:27 +08:00
updatedPen, err = s.penRepository.GetPenByIDTx(tx, id)
2025-10-04 01:28:05 +08:00
if err != nil {
s.logger.Errorf("更新猪栏状态后获取猪栏 %d 信息失败: %v", id, err)
return fmt.Errorf("更新猪栏状态后获取猪栏 %d 信息失败: %w", id, err)
}
return nil
})
if err != nil {
return nil, err
}
return updatedPen, nil
}