zero构建

This commit is contained in:
2025-10-05 18:28:59 +08:00
parent 934f77323c
commit 1e6c2e88a1
19 changed files with 81 additions and 503 deletions

View File

@@ -0,0 +1,27 @@
// Code scaffolded by goctl. Safe to edit.
// goctl 1.9.1
package upload
import (
"github.com/JACKYMYPERSON/hldrCenter/config"
"github.com/JACKYMYPERSON/hldrCenter/internal/upload/internal/logic/upload"
"github.com/gin-gonic/gin"
)
func UploadImageHandler(cfg *config.Config) gin.HandlerFunc {
return func(c *gin.Context) {
// 1. 传递上下文:直接用 Gin 上下文的 c.Context()(包含超时、取消等信息,比 c.Request.Context() 更完整)
ctx := c.Request.Context()
// 2. 创建 Logic 实例:参数顺序/类型与 Logic 层构造函数完全匹配ctx, cfg, c
// 注意:此处第三个参数是 Gin 上下文 c与你 Logic 层的 `c *gin.Context` 字段对应
l := upload.NewUploadImageLogic(ctx, cfg, c)
// 3. 调用 Logic 层方法:无需处理返回的 resp/err因为 Logic 层已通过 l.c.JSON 返回响应)
// 即使返回 errLogic 层也已提前返回错误响应,此处仅需调用方法即可
_, _ = l.UploadImage()
// 4. 无需额外响应处理!!!(关键:避免重复调用 c.JSON/httpx 导致错误)
}
}

View File

@@ -1,24 +0,0 @@
// Code scaffolded by goctl. Safe to edit.
// goctl 1.9.1
package upload
import (
"net/http"
"github.com/JACKYMYPERSON/hldrCenter/internal/upload/internal/logic/upload"
"github.com/JACKYMYPERSON/hldrCenter/internal/upload/internal/svc"
"github.com/zeromicro/go-zero/rest/httpx"
)
func UploadImageHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
l := upload.NewUploadImageLogic(r.Context(), svcCtx)
resp, err := l.UploadImage()
if err != nil {
httpx.ErrorCtx(r.Context(), w, err)
} else {
httpx.OkJsonCtx(r.Context(), w, resp)
}
}
}

View File

@@ -10,7 +10,7 @@ import (
"strings"
"time"
"github.com/JACKYMYPERSON/hldrCenter/internal/upload/internal/svc"
"github.com/JACKYMYPERSON/hldrCenter/config"
"github.com/JACKYMYPERSON/hldrCenter/internal/upload/internal/types"
"github.com/aliyun/aliyun-oss-go-sdk/oss"
"github.com/gin-gonic/gin"
@@ -20,23 +20,25 @@ import (
type UploadImageLogic struct {
logx.Logger
ctx context.Context
svcCtx *svc.ServiceContext
ctx context.Context
cfg *config.Config
c *gin.Context
}
func NewUploadImageLogic(ctx context.Context, svcCtx *svc.ServiceContext) *UploadImageLogic {
func NewUploadImageLogic(ctx context.Context, cfg *config.Config, c *gin.Context) *UploadImageLogic {
return &UploadImageLogic{
Logger: logx.WithContext(ctx),
ctx: ctx,
svcCtx: svcCtx,
cfg: cfg,
c: c,
}
}
func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error) {
// 获取上传的图片文件
fileHeader, err := c.FormFile("image")
fileHeader, err := l.c.FormFile("image")
if err != nil {
c.JSON(http.StatusBadRequest, gin.H{
l.c.JSON(http.StatusBadRequest, gin.H{
"code": 400,
"message": "获取图片失败,请重新上传",
"error": err.Error(),
@@ -47,7 +49,7 @@ func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error
// 打开文件
file, err := fileHeader.Open()
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{
l.c.JSON(http.StatusInternalServerError, gin.H{
"code": 500,
"message": "打开图片文件失败",
"error": err.Error(),
@@ -57,32 +59,32 @@ func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error
defer file.Close()
// 检查文件大小使用配置中的max_file_size
if fileHeader.Size > cfg.Upload.MaxFileSize {
c.JSON(http.StatusBadRequest, gin.H{
if fileHeader.Size > l.cfg.Upload.MaxFileSize {
l.c.JSON(http.StatusBadRequest, gin.H{
"code": 400,
"message": fmt.Sprintf("图片大小不能超过 %dMB", cfg.Upload.MaxFileSize>>20),
"message": fmt.Sprintf("图片大小不能超过 %dMB", l.cfg.Upload.MaxFileSize>>20),
})
return
}
// 检查文件类型使用配置中的allow_image_types
fileType := fileHeader.Header.Get("Content-Type")
if !strings.Contains(cfg.Upload.AllowImageTypes, fileType) {
c.JSON(http.StatusBadRequest, gin.H{
if !strings.Contains(l.cfg.Upload.AllowImageTypes, fileType) {
l.c.JSON(http.StatusBadRequest, gin.H{
"code": 400,
"message": fmt.Sprintf("不支持的图片类型,仅允许:%s", cfg.Upload.AllowImageTypes),
"message": fmt.Sprintf("不支持的图片类型,仅允许:%s", l.cfg.Upload.AllowImageTypes),
})
return
}
// 初始化OSS客户端使用配置中的OSS参数
client, err := oss.New(
cfg.OSS.Endpoint,
cfg.OSS.AccessKeyID,
cfg.OSS.AccessKeySecret,
l.cfg.OSS.Endpoint,
l.cfg.OSS.AccessKeyID,
l.cfg.OSS.AccessKeySecret,
)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{
l.c.JSON(http.StatusInternalServerError, gin.H{
"code": 500,
"message": "初始化OSS客户端失败",
"error": err.Error(),
@@ -91,9 +93,9 @@ func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error
}
// 获取Bucket
bucket, err := client.Bucket(cfg.OSS.BucketName)
bucket, err := client.Bucket(l.cfg.OSS.BucketName)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{
l.c.JSON(http.StatusInternalServerError, gin.H{
"code": 500,
"message": "获取Bucket失败",
"error": err.Error(),
@@ -104,12 +106,12 @@ func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error
// 生成唯一文件名
timestamp := time.Now().Format("20060102150405")
filename := fmt.Sprintf("%s_%s", timestamp, fileHeader.Filename)
objectKey := cfg.OSS.ObjectPrefix + filename // OSS中的完整对象键
objectKey := l.cfg.OSS.ObjectPrefix + filename // OSS中的完整对象键
// 上传文件到OSS
err = bucket.PutObject(objectKey, file)
if err != nil {
c.JSON(http.StatusInternalServerError, gin.H{
l.c.JSON(http.StatusInternalServerError, gin.H{
"code": 500,
"message": "上传图片到OSS失败",
"error": err.Error(),
@@ -118,11 +120,11 @@ func (l *UploadImageLogic) UploadImage() (resp *types.UploadImageResp, err error
}
// 生成图片访问URL
host := strings.TrimPrefix(cfg.OSS.Endpoint, "https://")
imageURL := fmt.Sprintf("https://%s.%s/%s", cfg.OSS.BucketName, host, objectKey)
host := strings.TrimPrefix(l.cfg.OSS.Endpoint, "https://")
imageURL := fmt.Sprintf("https://%s.%s/%s", l.cfg.OSS.BucketName, host, objectKey)
// 返回成功响应
c.JSON(http.StatusOK, gin.H{
l.c.JSON(http.StatusOK, gin.H{
"code": 200,
"message": "图片上传成功",
"data": gin.H{"url": imageURL},