You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

package.go 6.5 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  1. package http
  2. import (
  3. "io"
  4. "mime/multipart"
  5. "net/http"
  6. "time"
  7. "github.com/gin-gonic/gin"
  8. "gitlink.org.cn/cloudream/common/consts/errorcode"
  9. "gitlink.org.cn/cloudream/common/models"
  10. "gitlink.org.cn/cloudream/common/pkgs/logger"
  11. "gitlink.org.cn/cloudream/common/utils/serder"
  12. "gitlink.org.cn/cloudream/storage-common/pkgs/iterator"
  13. )
  14. type PackageService struct {
  15. *Server
  16. }
  17. func (s *Server) PackageSvc() *PackageService {
  18. return &PackageService{
  19. Server: s,
  20. }
  21. }
  22. type PackageDownloadReq struct {
  23. UserID *int64 `form:"userID" binding:"required"`
  24. PackageID *int64 `form:"packageID" binding:"required"`
  25. }
  26. func (s *PackageService) Download(ctx *gin.Context) {
  27. log := logger.WithField("HTTP", "Package.Download")
  28. var req PackageDownloadReq
  29. if err := ctx.ShouldBindQuery(&req); err != nil {
  30. log.Warnf("binding body: %s", err.Error())
  31. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  32. return
  33. }
  34. file, err := s.svc.PackageSvc().DownloadPackage(*req.UserID, *req.PackageID)
  35. if err != nil {
  36. log.Warnf("downloading package: %s", err.Error())
  37. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "download package failed"))
  38. return
  39. }
  40. ctx.Writer.WriteHeader(http.StatusOK)
  41. // TODO 需要设置FileName
  42. ctx.Header("Content-Disposition", "attachment; filename=filename")
  43. ctx.Header("Content-Type", "application/octet-stream")
  44. buf := make([]byte, 4096)
  45. ctx.Stream(func(w io.Writer) bool {
  46. rd, err := file.Read(buf)
  47. if err == io.EOF {
  48. return false
  49. }
  50. if err != nil {
  51. log.Warnf("reading file data: %s", err.Error())
  52. return false
  53. }
  54. err = myio.WriteAll(w, buf[:rd])
  55. if err != nil {
  56. log.Warnf("writing data to response: %s", err.Error())
  57. return false
  58. }
  59. return true
  60. })
  61. }
  62. type PackageUploadReq struct {
  63. Info PackageUploadInfo `form:"info" binding:"required"`
  64. Files []*multipart.FileHeader `form:"files"`
  65. }
  66. type PackageUploadInfo struct {
  67. UserID *int64 `json:"userID" binding:"required"`
  68. BucketID *int64 `json:"bucketID" binding:"required"`
  69. Name string `json:"name" binding:"required"`
  70. Redundancy models.TypedRedundancyInfo `json:"redundancy" binding:"required"`
  71. }
  72. type PackageUploadResp struct {
  73. PackageID int64 `json:"packageID,string"`
  74. }
  75. func (s *PackageService) Upload(ctx *gin.Context) {
  76. log := logger.WithField("HTTP", "Package.Upload")
  77. var req PackageUploadReq
  78. if err := ctx.ShouldBind(&req); err != nil {
  79. log.Warnf("binding body: %s", err.Error())
  80. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  81. return
  82. }
  83. switch req.Info.Redundancy.Type {
  84. case models.RedundancyRep:
  85. s.uploadRep(ctx, &req)
  86. return
  87. case models.RedundancyEC:
  88. s.uploadEC(ctx, &req)
  89. return
  90. }
  91. ctx.JSON(http.StatusForbidden, Failed(errorcode.OperationFailed, "not supported yet"))
  92. }
  93. func (s *PackageService) uploadRep(ctx *gin.Context, req *PackageUploadReq) {
  94. log := logger.WithField("HTTP", "Package.Upload")
  95. var repInfo models.RepRedundancyInfo
  96. if err := serder.AnyToAny(req.Info.Redundancy.Info, &repInfo); err != nil {
  97. log.Warnf("parsing rep redundancy config: %s", err.Error())
  98. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "invalid rep redundancy config"))
  99. return
  100. }
  101. objIter := iterator.NewHTTPObjectIterator(req.Files)
  102. taskID, err := s.svc.PackageSvc().StartCreatingRepPackage(*req.Info.UserID, *req.Info.BucketID, req.Info.Name, objIter, repInfo)
  103. if err != nil {
  104. log.Warnf("start uploading rep package task: %s", err.Error())
  105. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "start uploading task failed"))
  106. return
  107. }
  108. for {
  109. complete, createResult, err := s.svc.PackageSvc().WaitCreatingRepPackage(taskID, time.Second*5)
  110. if complete {
  111. if err != nil {
  112. log.Warnf("uploading rep package: %s", err.Error())
  113. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "uploading rep package failed"))
  114. return
  115. }
  116. ctx.JSON(http.StatusOK, OK(PackageUploadResp{
  117. PackageID: createResult.PackageID,
  118. }))
  119. return
  120. }
  121. if err != nil {
  122. log.Warnf("waiting task: %s", err.Error())
  123. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "wait uploading task failed"))
  124. return
  125. }
  126. }
  127. }
  128. func (s *PackageService) uploadEC(ctx *gin.Context, req *PackageUploadReq) {
  129. log := logger.WithField("HTTP", "Package.Upload")
  130. var ecInfo models.ECRedundancyInfo
  131. if err := serder.AnyToAny(req.Info.Redundancy.Info, &ecInfo); err != nil {
  132. log.Warnf("parsing ec redundancy config: %s", err.Error())
  133. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "invalid rep redundancy config"))
  134. return
  135. }
  136. objIter := iterator.NewHTTPObjectIterator(req.Files)
  137. taskID, err := s.svc.PackageSvc().StartCreatingECPackage(*req.Info.UserID, *req.Info.BucketID, req.Info.Name, objIter, ecInfo)
  138. if err != nil {
  139. log.Warnf("start uploading ec package task: %s", err.Error())
  140. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "start uploading task failed"))
  141. return
  142. }
  143. for {
  144. complete, createResult, err := s.svc.PackageSvc().WaitCreatingECPackage(taskID, time.Second*5)
  145. if complete {
  146. if err != nil {
  147. log.Warnf("uploading ec package: %s", err.Error())
  148. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "uploading ec package failed"))
  149. return
  150. }
  151. ctx.JSON(http.StatusOK, OK(PackageUploadResp{
  152. PackageID: createResult.PackageID,
  153. }))
  154. return
  155. }
  156. if err != nil {
  157. log.Warnf("waiting task: %s", err.Error())
  158. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "wait uploading task failed"))
  159. return
  160. }
  161. }
  162. }
  163. type PackageDeleteReq struct {
  164. UserID *int64 `json:"userID" binding:"required"`
  165. PackageID *int64 `json:"packageID" binding:"required"`
  166. }
  167. func (s *PackageService) Delete(ctx *gin.Context) {
  168. log := logger.WithField("HTTP", "Package.Delete")
  169. var req PackageDeleteReq
  170. if err := ctx.ShouldBindJSON(&req); err != nil {
  171. log.Warnf("binding body: %s", err.Error())
  172. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  173. return
  174. }
  175. err := s.svc.PackageSvc().DeletePackage(*req.UserID, *req.PackageID)
  176. if err != nil {
  177. log.Warnf("deleting package: %s", err.Error())
  178. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "delete package failed"))
  179. return
  180. }
  181. ctx.JSON(http.StatusOK, OK(nil))
  182. }

本项目旨在将云际存储公共基础设施化,使个人及企业可低门槛使用高效的云际存储服务(安装开箱即用云际存储客户端即可,无需关注其他组件的部署),同时支持用户灵活便捷定制云际存储的功能细节。