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 7.5 kB

2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. package http
  2. import (
  3. "mime/multipart"
  4. "net/http"
  5. "time"
  6. "github.com/gin-gonic/gin"
  7. "gitlink.org.cn/cloudream/common/consts/errorcode"
  8. "gitlink.org.cn/cloudream/common/pkgs/iterator"
  9. "gitlink.org.cn/cloudream/common/pkgs/logger"
  10. stgsdk "gitlink.org.cn/cloudream/common/sdks/storage"
  11. stgiter "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  12. )
  13. type PackageService struct {
  14. *Server
  15. }
  16. func (s *Server) PackageSvc() *PackageService {
  17. return &PackageService{
  18. Server: s,
  19. }
  20. }
  21. type PackageUploadReq struct {
  22. Info PackageUploadInfo `form:"info" binding:"required"`
  23. Files []*multipart.FileHeader `form:"files"`
  24. }
  25. type PackageUploadInfo struct {
  26. UserID *int64 `json:"userID" binding:"required"`
  27. BucketID *int64 `json:"bucketID" binding:"required"`
  28. Name string `json:"name" binding:"required"`
  29. Redundancy stgsdk.TypedRedundancyInfo `json:"redundancy" binding:"required"`
  30. NodeAffinity *int64 `json:"nodeAffinity"`
  31. }
  32. type PackageUploadResp struct {
  33. PackageID int64 `json:"packageID,string"`
  34. }
  35. func (s *PackageService) Upload(ctx *gin.Context) {
  36. log := logger.WithField("HTTP", "Package.Upload")
  37. var req PackageUploadReq
  38. if err := ctx.ShouldBind(&req); err != nil {
  39. log.Warnf("binding body: %s", err.Error())
  40. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  41. return
  42. }
  43. if req.Info.Redundancy.IsRepInfo() {
  44. s.uploadRep(ctx, &req)
  45. return
  46. }
  47. if req.Info.Redundancy.IsECInfo() {
  48. s.uploadEC(ctx, &req)
  49. return
  50. }
  51. ctx.JSON(http.StatusForbidden, Failed(errorcode.OperationFailed, "not supported yet"))
  52. }
  53. func (s *PackageService) uploadRep(ctx *gin.Context, req *PackageUploadReq) {
  54. log := logger.WithField("HTTP", "Package.Upload")
  55. var err error
  56. var repInfo stgsdk.RepRedundancyInfo
  57. if repInfo, err = req.Info.Redundancy.ToRepInfo(); err != nil {
  58. log.Warnf("parsing rep redundancy config: %s", err.Error())
  59. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "invalid rep redundancy config"))
  60. return
  61. }
  62. objIter := mapMultiPartFileToUploadingObject(req.Files)
  63. taskID, err := s.svc.PackageSvc().StartCreatingRepPackage(*req.Info.UserID, *req.Info.BucketID, req.Info.Name, objIter, repInfo, req.Info.NodeAffinity)
  64. if err != nil {
  65. log.Warnf("start uploading rep package task: %s", err.Error())
  66. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "start uploading task failed"))
  67. return
  68. }
  69. for {
  70. complete, createResult, err := s.svc.PackageSvc().WaitCreatingRepPackage(taskID, time.Second*5)
  71. if complete {
  72. if err != nil {
  73. log.Warnf("uploading rep package: %s", err.Error())
  74. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "uploading rep package failed"))
  75. return
  76. }
  77. ctx.JSON(http.StatusOK, OK(PackageUploadResp{
  78. PackageID: createResult.PackageID,
  79. }))
  80. return
  81. }
  82. if err != nil {
  83. log.Warnf("waiting task: %s", err.Error())
  84. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "wait uploading task failed"))
  85. return
  86. }
  87. }
  88. }
  89. func (s *PackageService) uploadEC(ctx *gin.Context, req *PackageUploadReq) {
  90. log := logger.WithField("HTTP", "Package.Upload")
  91. var err error
  92. var ecInfo stgsdk.ECRedundancyInfo
  93. if ecInfo, err = req.Info.Redundancy.ToECInfo(); err != nil {
  94. log.Warnf("parsing ec redundancy config: %s", err.Error())
  95. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "invalid rep redundancy config"))
  96. return
  97. }
  98. objIter := mapMultiPartFileToUploadingObject(req.Files)
  99. taskID, err := s.svc.PackageSvc().StartCreatingECPackage(*req.Info.UserID, *req.Info.BucketID, req.Info.Name, objIter, ecInfo, req.Info.NodeAffinity)
  100. if err != nil {
  101. log.Warnf("start uploading ec package task: %s", err.Error())
  102. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "start uploading task failed"))
  103. return
  104. }
  105. for {
  106. complete, createResult, err := s.svc.PackageSvc().WaitCreatingECPackage(taskID, time.Second*5)
  107. if complete {
  108. if err != nil {
  109. log.Warnf("uploading ec package: %s", err.Error())
  110. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "uploading ec package failed"))
  111. return
  112. }
  113. ctx.JSON(http.StatusOK, OK(PackageUploadResp{
  114. PackageID: createResult.PackageID,
  115. }))
  116. return
  117. }
  118. if err != nil {
  119. log.Warnf("waiting task: %s", err.Error())
  120. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "wait uploading task failed"))
  121. return
  122. }
  123. }
  124. }
  125. type PackageDeleteReq struct {
  126. UserID *int64 `json:"userID" binding:"required"`
  127. PackageID *int64 `json:"packageID" binding:"required"`
  128. }
  129. func (s *PackageService) Delete(ctx *gin.Context) {
  130. log := logger.WithField("HTTP", "Package.Delete")
  131. var req PackageDeleteReq
  132. if err := ctx.ShouldBindJSON(&req); err != nil {
  133. log.Warnf("binding body: %s", err.Error())
  134. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  135. return
  136. }
  137. err := s.svc.PackageSvc().DeletePackage(*req.UserID, *req.PackageID)
  138. if err != nil {
  139. log.Warnf("deleting package: %s", err.Error())
  140. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "delete package failed"))
  141. return
  142. }
  143. ctx.JSON(http.StatusOK, OK(nil))
  144. }
  145. type GetCachedNodesReq struct {
  146. UserID *int64 `json:"userID" binding:"required"`
  147. PackageID *int64 `json:"packageID" binding:"required"`
  148. }
  149. type GetCachedNodesResp struct {
  150. stgsdk.PackageCachingInfo
  151. }
  152. func (s *PackageService) GetCachedNodes(ctx *gin.Context) {
  153. log := logger.WithField("HTTP", "Package.GetCachedNodes")
  154. var req GetCachedNodesReq
  155. if err := ctx.ShouldBindJSON(&req); err != nil {
  156. log.Warnf("binding body: %s", err.Error())
  157. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  158. return
  159. }
  160. resp, err := s.svc.PackageSvc().GetCachedNodes(*req.UserID, *req.PackageID)
  161. if err != nil {
  162. log.Warnf("get package cached nodes failed: %s", err.Error())
  163. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "get package cached nodes failed"))
  164. return
  165. }
  166. ctx.JSON(http.StatusOK, OK(GetCachedNodesResp{resp}))
  167. }
  168. type GetLoadedNodesReq struct {
  169. UserID *int64 `json:"userID" binding:"required"`
  170. PackageID *int64 `json:"packageID" binding:"required"`
  171. }
  172. type GetLoadedNodesResp struct {
  173. NodeIDs []int64 `json:"nodeIDs"`
  174. }
  175. func (s *PackageService) GetLoadedNodes(ctx *gin.Context) {
  176. log := logger.WithField("HTTP", "Package.GetLoadedNodes")
  177. var req GetLoadedNodesReq
  178. if err := ctx.ShouldBindJSON(&req); err != nil {
  179. log.Warnf("binding body: %s", err.Error())
  180. ctx.JSON(http.StatusBadRequest, Failed(errorcode.BadArgument, "missing argument or invalid argument"))
  181. return
  182. }
  183. nodeIDs, err := s.svc.PackageSvc().GetLoadedNodes(*req.UserID, *req.PackageID)
  184. if err != nil {
  185. log.Warnf("get package loaded nodes failed: %s", err.Error())
  186. ctx.JSON(http.StatusOK, Failed(errorcode.OperationFailed, "get package loaded nodes failed"))
  187. return
  188. }
  189. ctx.JSON(http.StatusOK, OK(GetLoadedNodesResp{
  190. NodeIDs: nodeIDs,
  191. }))
  192. }
  193. func mapMultiPartFileToUploadingObject(files []*multipart.FileHeader) stgiter.UploadingObjectIterator {
  194. return iterator.Map[*multipart.FileHeader](
  195. iterator.Array(files...),
  196. func(file *multipart.FileHeader) (*stgiter.IterUploadingObject, error) {
  197. stream, err := file.Open()
  198. if err != nil {
  199. return nil, err
  200. }
  201. return &stgiter.IterUploadingObject{
  202. Path: file.Filename,
  203. Size: file.Size,
  204. File: stream,
  205. }, nil
  206. },
  207. )
  208. }

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