package http import ( "io" "mime/multipart" "net/http" "time" "github.com/gin-gonic/gin" "gitlink.org.cn/cloudream/common/consts/errorcode" "gitlink.org.cn/cloudream/common/models" "gitlink.org.cn/cloudream/common/pkgs/logger" myio "gitlink.org.cn/cloudream/common/utils/io" "gitlink.org.cn/cloudream/common/utils/serder" "gitlink.org.cn/cloudream/storage-client/internal/task" ) type ObjectService struct { *Server } func (s *Server) ObjectSvc() *ObjectService { return &ObjectService{ Server: s, } } type ObjectDownloadReq struct { UserID *int64 `form:"userID" binding:"required"` ObjectID *int64 `form:"objectID" binding:"required"` } func (s *ObjectService) Download(ctx *gin.Context) { log := logger.WithField("HTTP", "Object.Download") var req ObjectDownloadReq if err := ctx.ShouldBindQuery(&req); err != nil { log.Warnf("binding body: %s", err.Error()) ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument")) return } file, err := s.svc.ObjectSvc().DownloadObject(*req.UserID, *req.ObjectID) if err != nil { log.Warnf("downloading object: %s", err.Error()) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "download object failed")) return } ctx.Writer.WriteHeader(http.StatusOK) // TODO 需要设置FileName ctx.Header("Content-Disposition", "attachment; filename=filename") ctx.Header("Content-Type", "application/octet-stream") buf := make([]byte, 4096) ctx.Stream(func(w io.Writer) bool { rd, err := file.Read(buf) if err == io.EOF { return false } if err != nil { log.Warnf("reading file data: %s", err.Error()) return false } err = myio.WriteAll(w, buf[:rd]) if err != nil { log.Warnf("writing data to response: %s", err.Error()) return false } return true }) } type ObjectUploadReq struct { Info ObjectUploadInfo `form:"info" binding:"required"` File *multipart.FileHeader `form:"file"` } type ObjectUploadInfo struct { UserID *int64 `json:"userID" binding:"required"` BucketID *int64 `json:"bucketID" binding:"required"` FileSize *int64 `json:"fileSize" binding:"required"` ObjectName string `json:"objectName" binding:"required"` Redundancy struct { Type string `json:"type" binding:"required"` Config any `json:"config" binding:"required"` } `json:"redundancy" binding:"required"` } type ObjectUploadResp struct { ObjectID int64 `json:"objectID,string"` } func (s *ObjectService) Upload(ctx *gin.Context) { log := logger.WithField("HTTP", "Object.Upload") var req ObjectUploadReq if err := ctx.ShouldBind(&req); err != nil { log.Warnf("binding body: %s", err.Error()) ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument")) return } switch req.Info.Redundancy.Type { case models.RedundancyRep: s.uploadRep(ctx, &req) return case models.RedundancyEC: } ctx.JSON(http.StatusForbidden, Failed(errorcode.OperationFailed, "not supported yet")) } func (s *ObjectService) uploadRep(ctx *gin.Context, req *ObjectUploadReq) { log := logger.WithField("HTTP", "Object.Upload") var repInfo models.RepRedundancyInfo if err := serder.AnyToAny(req.Info.Redundancy.Config, &repInfo); err != nil { log.Warnf("parsing rep redundancy config: %s", err.Error()) ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "invalid rep redundancy config")) return } file, err := req.File.Open() if err != nil { log.Warnf("opening file: %s", err.Error()) ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "open file failed")) return } taskID, err := s.svc.ObjectSvc().StartUploadingRepObjects(*req.Info.UserID, *req.Info.BucketID, []task.UploadObject{{ ObjectName: req.Info.ObjectName, File: file, FileSize: *req.Info.FileSize, }}, repInfo.RepCount) if err != nil { log.Warnf("start uploading rep object task: %s", err.Error()) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "start uploading task failed")) return } for { complete, uploadObjectResult, err := s.svc.ObjectSvc().WaitUploadingRepObjects(taskID, time.Second*5) if complete { if err != nil { log.Warnf("uploading rep object: %s", err.Error()) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "uploading rep object failed")) return } uploadRet := uploadObjectResult.Results[0] if uploadRet.Error != nil { log.Warnf("uploading rep object: %s", uploadRet.Error) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, uploadRet.Error.Error())) return } ctx.JSON(http.StatusOK, OK(ObjectUploadResp{ ObjectID: uploadRet.ObjectID, })) return } if err != nil { log.Warnf("waiting task: %s", err.Error()) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "wait uploading task failed")) return } } } type ObjectDeleteReq struct { UserID *int64 `json:"userID" binding:"required"` ObjectID *int64 `json:"objectID" binding:"required"` } func (s *ObjectService) Delete(ctx *gin.Context) { log := logger.WithField("HTTP", "Object.Delete") var req ObjectDeleteReq if err := ctx.ShouldBindJSON(&req); err != nil { log.Warnf("binding body: %s", err.Error()) ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument")) return } err := s.svc.ObjectSvc().DeleteObject(*req.UserID, *req.ObjectID) if err != nil { log.Warnf("deleting object: %s", err.Error()) ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "delete object failed")) return } ctx.JSON(http.StatusOK, OK(nil)) }