Files
pig-farm-controller/internal/app/controller/management/controller_helpers.go

236 lines
9.1 KiB
Go
Raw Normal View History

2025-10-06 23:48:31 +08:00
package management
import (
2025-11-05 23:10:51 +08:00
"context"
2025-10-06 23:48:31 +08:00
"errors"
2025-10-23 11:52:08 +08:00
"fmt"
2025-10-06 23:48:31 +08:00
"strconv"
"git.huangwc.com/pig/pig-farm-controller/internal/app/controller"
"git.huangwc.com/pig/pig-farm-controller/internal/app/service"
2025-11-05 23:10:51 +08:00
"git.huangwc.com/pig/pig-farm-controller/internal/infra/logs"
2025-11-05 19:57:30 +08:00
2025-10-30 17:15:14 +08:00
"github.com/labstack/echo/v4"
2025-10-06 23:48:31 +08:00
)
// mapAndSendError 统一映射服务层错误并发送响应。
// 这个函数将服务层返回的错误转换为控制器层应返回的HTTP状态码和审计信息。
2025-11-05 23:10:51 +08:00
func mapAndSendError(reqContext context.Context, c *PigBatchController, ctx echo.Context, action string, err error, id uint) error {
2025-10-06 23:48:31 +08:00
if errors.Is(err, service.ErrPigBatchNotFound) ||
errors.Is(err, service.ErrPenNotFound) ||
errors.Is(err, service.ErrPenNotAssociatedWithBatch) {
2025-10-30 17:15:14 +08:00
return controller.SendErrorWithAudit(ctx, controller.CodeNotFound, err.Error(), action, err.Error(), id)
2025-10-06 23:48:31 +08:00
} else if errors.Is(err, service.ErrInvalidOperation) ||
errors.Is(err, service.ErrPigBatchActive) ||
errors.Is(err, service.ErrPigBatchNotActive) ||
errors.Is(err, service.ErrPenOccupiedByOtherBatch) ||
errors.Is(err, service.ErrPenStatusInvalidForAllocation) ||
errors.Is(err, service.ErrPenNotEmpty) {
2025-10-30 17:15:14 +08:00
return controller.SendErrorWithAudit(ctx, controller.CodeConflict, err.Error(), action, err.Error(), id)
2025-10-06 23:48:31 +08:00
} else {
2025-11-05 23:10:51 +08:00
logs.GetLogger(reqContext).Errorf("操作[%s]业务逻辑失败: %v", action, err)
2025-10-30 17:15:14 +08:00
return controller.SendErrorWithAudit(ctx, controller.CodeInternalError, fmt.Sprintf("操作失败: %v", err), action, err.Error(), id)
2025-10-06 23:48:31 +08:00
}
}
2025-10-30 17:15:14 +08:00
// idExtractorFunc 定义了一个函数类型用于从echo.Context中提取主ID。
type idExtractorFunc func(ctx echo.Context) (uint, error)
2025-10-06 23:48:31 +08:00
2025-10-30 17:15:14 +08:00
// extractOperatorAndPrimaryID 封装了从echo.Context中提取操作员ID和主ID的通用逻辑。
2025-10-07 00:18:17 +08:00
// 它负责处理ID提取过程中的错误并发送相应的HTTP响应。
2025-10-06 23:48:31 +08:00
//
// 参数:
//
2025-10-07 00:18:17 +08:00
// c: *PigBatchController - 控制器实例,用于访问其日志。
2025-10-30 17:15:14 +08:00
// ctx: echo.Context - Echo上下文。
2025-10-06 23:48:31 +08:00
// action: string - 当前操作的描述,用于日志和审计。
// idExtractor: idExtractorFunc - 可选函数用于从ctx中提取主ID。如果为nil则尝试从":id"路径参数中提取。
2025-10-07 00:18:17 +08:00
//
// 返回值:
//
// operatorID: uint - 提取到的操作员ID。
// primaryID: uint - 提取到的主ID。
2025-10-30 17:15:14 +08:00
// err: error - 如果ID提取失败或发送错误响应则返回错误。
2025-10-07 00:18:17 +08:00
func extractOperatorAndPrimaryID(
2025-10-06 23:48:31 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-06 23:48:31 +08:00
action string,
idExtractor idExtractorFunc,
2025-10-30 17:15:14 +08:00
) (operatorID uint, primaryID uint, err error) {
2025-10-07 00:18:17 +08:00
// 1. 获取操作员ID
2025-10-30 17:15:14 +08:00
operatorID, err = controller.GetOperatorIDFromContext(ctx)
2025-10-06 23:48:31 +08:00
if err != nil {
2025-10-30 17:15:14 +08:00
return 0, 0, controller.SendErrorWithAudit(ctx, controller.CodeUnauthorized, "未授权", action, "无法获取操作员ID", nil)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 2. 提取主ID
2025-10-06 23:48:31 +08:00
if idExtractor != nil {
primaryID, err = idExtractor(ctx)
if err != nil {
2025-10-30 17:15:14 +08:00
return 0, 0, controller.SendErrorWithAudit(ctx, controller.CodeBadRequest, "无效的ID格式", action, "ID格式错误", err.Error())
2025-10-06 23:48:31 +08:00
}
} else { // 默认从 ":id" 路径参数提取
idParam := ctx.Param("id")
if idParam == "" { // 有些端点可能没有 "id" 参数,例如列表或创建操作
// 如果没有ID参数且没有自定义提取器primaryID保持为0这对于某些操作是可接受的
} else {
parsedID, err := strconv.ParseUint(idParam, 10, 32)
if err != nil {
2025-10-30 17:15:14 +08:00
return 0, 0, controller.SendErrorWithAudit(ctx, controller.CodeBadRequest, "无效的ID格式", action, "ID格式错误", idParam)
2025-10-06 23:48:31 +08:00
}
primaryID = uint(parsedID)
}
}
2025-10-30 17:15:14 +08:00
return operatorID, primaryID, nil
2025-10-07 00:18:17 +08:00
}
// handleAPIRequest 封装了控制器中处理带有请求体和路径参数的API请求的通用逻辑。
// 它负责请求体绑定、操作员ID获取、服务层调用、错误映射和响应发送。
func handleAPIRequest[Req any](
2025-11-05 23:10:51 +08:00
reqContext context.Context,
2025-10-07 00:18:17 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-07 00:18:17 +08:00
action string,
reqDTO Req,
2025-10-30 17:15:14 +08:00
serviceExecutor func(ctx echo.Context, operatorID uint, primaryID uint, req Req) error,
2025-10-07 00:18:17 +08:00
successMsg string,
idExtractor idExtractorFunc,
2025-10-30 17:15:14 +08:00
) error {
2025-10-07 00:18:17 +08:00
// 1. 绑定请求体
2025-10-30 17:15:14 +08:00
if err := ctx.Bind(&reqDTO); err != nil {
return controller.SendErrorWithAudit(ctx, controller.CodeBadRequest, "无效的请求体", action, "请求体绑定失败", reqDTO)
2025-10-07 00:18:17 +08:00
}
// 2. 提取操作员ID和主ID
2025-10-30 17:15:14 +08:00
operatorID, primaryID, err := extractOperatorAndPrimaryID(c, ctx, action, idExtractor)
if err != nil {
return err // 错误已在 extractOperatorAndPrimaryID 中处理
2025-10-07 00:18:17 +08:00
}
// 3. 执行服务层逻辑
2025-10-30 17:15:14 +08:00
err = serviceExecutor(ctx, operatorID, primaryID, reqDTO)
2025-10-06 23:48:31 +08:00
if err != nil {
2025-11-05 23:10:51 +08:00
return mapAndSendError(reqContext, c, ctx, action, err, primaryID)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 4. 发送成功响应
2025-10-30 17:15:14 +08:00
return controller.SendSuccessWithAudit(ctx, controller.CodeSuccess, successMsg, nil, action, successMsg, primaryID)
2025-10-06 23:48:31 +08:00
}
// handleNoBodyAPIRequest 封装了处理不带请求体但有路径参数和操作员ID的API请求的通用逻辑。
func handleNoBodyAPIRequest(
2025-11-05 23:10:51 +08:00
reqContext context.Context,
2025-10-06 23:48:31 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-06 23:48:31 +08:00
action string,
2025-10-30 17:15:14 +08:00
serviceExecutor func(ctx echo.Context, operatorID uint, primaryID uint) error,
2025-10-06 23:48:31 +08:00
successMsg string,
idExtractor idExtractorFunc,
2025-10-30 17:15:14 +08:00
) error {
2025-10-07 00:18:17 +08:00
// 1. 提取操作员ID和主ID
2025-10-30 17:15:14 +08:00
operatorID, primaryID, err := extractOperatorAndPrimaryID(c, ctx, action, idExtractor)
if err != nil {
return err // 错误已在 extractOperatorAndPrimaryID 中处理
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 2. 执行服务层逻辑
2025-10-30 17:15:14 +08:00
err = serviceExecutor(ctx, operatorID, primaryID)
2025-10-06 23:48:31 +08:00
if err != nil {
2025-11-05 23:10:51 +08:00
return mapAndSendError(reqContext, c, ctx, action, err, primaryID)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 3. 发送成功响应
2025-10-30 17:15:14 +08:00
return controller.SendSuccessWithAudit(ctx, controller.CodeSuccess, successMsg, nil, action, successMsg, primaryID)
2025-10-06 23:48:31 +08:00
}
// handleAPIRequestWithResponse 封装了控制器中处理带有请求体、路径参数并返回响应DTO的API请求的通用逻辑。
func handleAPIRequestWithResponse[Req any, Resp any](
2025-11-05 23:10:51 +08:00
reqContext context.Context,
2025-10-06 23:48:31 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-06 23:48:31 +08:00
action string,
reqDTO Req,
2025-10-30 17:15:14 +08:00
serviceExecutor func(ctx echo.Context, operatorID uint, primaryID uint, req Req) (Resp, error), // serviceExecutor现在返回Resp
2025-10-06 23:48:31 +08:00
successMsg string,
idExtractor idExtractorFunc,
2025-10-30 17:15:14 +08:00
) error {
2025-10-06 23:48:31 +08:00
// 1. 绑定请求体
2025-10-30 17:15:14 +08:00
if err := ctx.Bind(&reqDTO); err != nil {
return controller.SendErrorWithAudit(ctx, controller.CodeBadRequest, fmt.Sprintf("无效的请求体: %v", err), action, fmt.Sprintf("请求体绑定失败: %v", err), reqDTO)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 2. 提取操作员ID和主ID
2025-10-30 17:15:14 +08:00
operatorID, primaryID, err := extractOperatorAndPrimaryID(c, ctx, action, idExtractor)
if err != nil {
return err // 错误已在 extractOperatorAndPrimaryID 中处理
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 3. 执行服务层逻辑
2025-10-06 23:48:31 +08:00
respDTO, err := serviceExecutor(ctx, operatorID, primaryID, reqDTO)
if err != nil {
2025-11-05 23:10:51 +08:00
return mapAndSendError(reqContext, c, ctx, action, err, primaryID)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 4. 发送成功响应
2025-10-30 17:15:14 +08:00
return controller.SendSuccessWithAudit(ctx, controller.CodeSuccess, successMsg, respDTO, action, successMsg, primaryID)
2025-10-06 23:48:31 +08:00
}
// handleNoBodyAPIRequestWithResponse 封装了处理不带请求体但有路径参数和操作员ID并返回响应DTO的API请求的通用逻辑。
func handleNoBodyAPIRequestWithResponse[Resp any](
2025-11-05 23:10:51 +08:00
reqContext context.Context,
2025-10-06 23:48:31 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-06 23:48:31 +08:00
action string,
2025-10-30 17:15:14 +08:00
serviceExecutor func(ctx echo.Context, operatorID uint, primaryID uint) (Resp, error), // serviceExecutor现在返回Resp
2025-10-06 23:48:31 +08:00
successMsg string,
idExtractor idExtractorFunc,
2025-10-30 17:15:14 +08:00
) error {
2025-10-07 00:18:17 +08:00
// 1. 提取操作员ID和主ID
2025-10-30 17:15:14 +08:00
operatorID, primaryID, err := extractOperatorAndPrimaryID(c, ctx, action, idExtractor)
if err != nil {
return err // 错误已在 extractOperatorAndPrimaryID 中处理
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 2. 执行服务层逻辑
2025-10-06 23:48:31 +08:00
respDTO, err := serviceExecutor(ctx, operatorID, primaryID)
if err != nil {
2025-11-05 23:10:51 +08:00
return mapAndSendError(reqContext, c, ctx, action, err, primaryID)
2025-10-06 23:48:31 +08:00
}
2025-10-07 00:18:17 +08:00
// 3. 发送成功响应
2025-10-30 17:15:14 +08:00
return controller.SendSuccessWithAudit(ctx, controller.CodeSuccess, successMsg, respDTO, action, successMsg, primaryID)
2025-10-06 23:48:31 +08:00
}
// handleQueryAPIRequestWithResponse 封装了处理带有查询参数并返回响应DTO的API请求的通用逻辑。
func handleQueryAPIRequestWithResponse[Query any, Resp any](
2025-11-05 23:10:51 +08:00
reqContext context.Context,
2025-10-06 23:48:31 +08:00
c *PigBatchController,
2025-10-30 17:15:14 +08:00
ctx echo.Context,
2025-10-06 23:48:31 +08:00
action string,
queryDTO Query,
2025-10-30 17:15:14 +08:00
serviceExecutor func(ctx echo.Context, operatorID uint, query Query) (Resp, error), // serviceExecutor现在接收queryDTO
2025-10-06 23:48:31 +08:00
successMsg string,
2025-10-30 17:15:14 +08:00
) error {
2025-10-06 23:48:31 +08:00
// 1. 绑定查询参数
2025-10-30 17:15:14 +08:00
if err := ctx.Bind(&queryDTO); err != nil {
return controller.SendErrorWithAudit(ctx, controller.CodeBadRequest, "无效的查询参数", action, "查询参数绑定失败", queryDTO)
2025-10-06 23:48:31 +08:00
}
// 2. 获取操作员ID
operatorID, err := controller.GetOperatorIDFromContext(ctx)
if err != nil {
2025-10-30 17:15:14 +08:00
return controller.SendErrorWithAudit(ctx, controller.CodeUnauthorized, "未授权", action, "无法获取操作员ID", nil)
2025-10-06 23:48:31 +08:00
}
// 3. 执行服务层逻辑
respDTO, err := serviceExecutor(ctx, operatorID, queryDTO)
if err != nil {
// 对于列表查询通常没有primaryID所以传递0
2025-11-05 23:10:51 +08:00
return mapAndSendError(reqContext, c, ctx, action, err, 0)
2025-10-06 23:48:31 +08:00
}
// 4. 发送成功响应
2025-10-30 17:15:14 +08:00
return controller.SendSuccessWithAudit(ctx, controller.CodeSuccess, successMsg, respDTO, action, successMsg, nil)
2025-10-06 23:48:31 +08:00
}