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.

storage.go 8.4 kB

2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247
  1. package mq
  2. import (
  3. "io/fs"
  4. "os"
  5. "path/filepath"
  6. "time"
  7. "github.com/samber/lo"
  8. "gitlink.org.cn/cloudream/common/consts/errorcode"
  9. "gitlink.org.cn/cloudream/common/pkgs/logger"
  10. "gitlink.org.cn/cloudream/common/pkgs/mq"
  11. "gitlink.org.cn/cloudream/storage/agent/internal/config"
  12. mytask "gitlink.org.cn/cloudream/storage/agent/internal/task"
  13. "gitlink.org.cn/cloudream/storage/common/consts"
  14. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  15. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  16. agtmq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/agent"
  17. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  18. "gitlink.org.cn/cloudream/storage/common/utils"
  19. )
  20. func (svc *Service) StartStorageLoadPackage(msg *agtmq.StartStorageLoadPackage) (*agtmq.StartStorageLoadPackageResp, *mq.CodeMessage) {
  21. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  22. if err != nil {
  23. logger.Warnf("new coordinator client: %s", err.Error())
  24. return nil, mq.Failed(errorcode.OperationFailed, "new coordinator client failed")
  25. }
  26. defer stgglb.CoordinatorMQPool.Release(coorCli)
  27. getStgResp, err := coorCli.GetStorageInfo(coormq.NewGetStorageInfo(msg.UserID, msg.StorageID))
  28. if err != nil {
  29. logger.WithField("StorageID", msg.StorageID).
  30. Warnf("getting storage info: %s", err.Error())
  31. return nil, mq.Failed(errorcode.OperationFailed, "get storage info failed")
  32. }
  33. outputDirPath := filepath.Join(config.Cfg().StorageBaseDir, getStgResp.Directory, utils.MakeStorageLoadPackageDirName(msg.PackageID, msg.UserID))
  34. if err = os.MkdirAll(outputDirPath, 0755); err != nil {
  35. logger.WithField("StorageID", msg.StorageID).
  36. Warnf("creating output directory: %s", err.Error())
  37. return nil, mq.Failed(errorcode.OperationFailed, "create output directory failed")
  38. }
  39. tsk := svc.taskManager.StartNew(mytask.NewDownloadPackage(msg.UserID, msg.PackageID, outputDirPath))
  40. return mq.ReplyOK(agtmq.NewStartStorageLoadPackageResp(tsk.ID()))
  41. }
  42. func (svc *Service) WaitStorageLoadPackage(msg *agtmq.WaitStorageLoadPackage) (*agtmq.WaitStorageLoadPackageResp, *mq.CodeMessage) {
  43. logger.WithField("TaskID", msg.TaskID).Debugf("wait loading package")
  44. tsk := svc.taskManager.FindByID(msg.TaskID)
  45. if tsk == nil {
  46. return nil, mq.Failed(errorcode.TaskNotFound, "task not found")
  47. }
  48. if msg.WaitTimeoutMs == 0 {
  49. tsk.Wait()
  50. errMsg := ""
  51. if tsk.Error() != nil {
  52. errMsg = tsk.Error().Error()
  53. }
  54. return mq.ReplyOK(agtmq.NewWaitStorageLoadPackageResp(true, errMsg))
  55. } else {
  56. if tsk.WaitTimeout(time.Duration(msg.WaitTimeoutMs)) {
  57. errMsg := ""
  58. if tsk.Error() != nil {
  59. errMsg = tsk.Error().Error()
  60. }
  61. return mq.ReplyOK(agtmq.NewWaitStorageLoadPackageResp(true, errMsg))
  62. }
  63. return mq.ReplyOK(agtmq.NewWaitStorageLoadPackageResp(false, ""))
  64. }
  65. }
  66. func (svc *Service) StorageCheck(msg *agtmq.StorageCheck) (*agtmq.StorageCheckResp, *mq.CodeMessage) {
  67. dirFullPath := filepath.Join(config.Cfg().StorageBaseDir, msg.Directory)
  68. infos, err := os.ReadDir(dirFullPath)
  69. if err != nil {
  70. logger.Warnf("list storage directory failed, err: %s", err.Error())
  71. return mq.ReplyOK(agtmq.NewStorageCheckResp(
  72. err.Error(),
  73. nil,
  74. ))
  75. }
  76. dirInfos := lo.Filter(infos, func(info fs.DirEntry, index int) bool { return info.IsDir() })
  77. if msg.IsComplete {
  78. return svc.checkStorageComplete(msg, dirInfos)
  79. } else {
  80. return svc.checkStorageIncrement(msg, dirInfos)
  81. }
  82. }
  83. func (svc *Service) checkStorageIncrement(msg *agtmq.StorageCheck, dirInfos []fs.DirEntry) (*agtmq.StorageCheckResp, *mq.CodeMessage) {
  84. infosMap := make(map[string]fs.DirEntry)
  85. for _, info := range dirInfos {
  86. infosMap[info.Name()] = info
  87. }
  88. var entries []agtmq.StorageCheckRespEntry
  89. for _, obj := range msg.Packages {
  90. dirName := utils.MakeStorageLoadPackageDirName(obj.PackageID, obj.UserID)
  91. _, ok := infosMap[dirName]
  92. if ok {
  93. // 不需要做处理
  94. // 删除map中的记录,表示此记录已被检查过
  95. delete(infosMap, dirName)
  96. } else {
  97. // 只要文件不存在,就删除StoragePackage表中的记录
  98. entries = append(entries, agtmq.NewStorageCheckRespEntry(obj.PackageID, obj.UserID, agtmq.CHECK_STORAGE_RESP_OP_DELETE))
  99. }
  100. }
  101. // 增量情况下,不需要对infosMap中没检查的记录进行处理
  102. return mq.ReplyOK(agtmq.NewStorageCheckResp(consts.StorageDirectoryStateOK, entries))
  103. }
  104. func (svc *Service) checkStorageComplete(msg *agtmq.StorageCheck, dirInfos []fs.DirEntry) (*agtmq.StorageCheckResp, *mq.CodeMessage) {
  105. infosMap := make(map[string]fs.DirEntry)
  106. for _, info := range dirInfos {
  107. infosMap[info.Name()] = info
  108. }
  109. var entries []agtmq.StorageCheckRespEntry
  110. for _, obj := range msg.Packages {
  111. dirName := utils.MakeStorageLoadPackageDirName(obj.PackageID, obj.UserID)
  112. _, ok := infosMap[dirName]
  113. if ok {
  114. // 不需要做处理
  115. // 删除map中的记录,表示此记录已被检查过
  116. delete(infosMap, dirName)
  117. } else {
  118. // 只要文件不存在,就删除StoragePackage表中的记录
  119. entries = append(entries, agtmq.NewStorageCheckRespEntry(obj.PackageID, obj.UserID, agtmq.CHECK_STORAGE_RESP_OP_DELETE))
  120. }
  121. }
  122. return mq.ReplyOK(agtmq.NewStorageCheckResp(consts.StorageDirectoryStateOK, entries))
  123. }
  124. func (svc *Service) StartStorageCreatePackage(msg *agtmq.StartStorageCreatePackage) (*agtmq.StartStorageCreatePackageResp, *mq.CodeMessage) {
  125. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  126. if err != nil {
  127. logger.Warnf("new coordinator client: %s", err.Error())
  128. return nil, mq.Failed(errorcode.OperationFailed, "new coordinator client failed")
  129. }
  130. defer stgglb.CoordinatorMQPool.Release(coorCli)
  131. getStgResp, err := coorCli.GetStorageInfo(coormq.NewGetStorageInfo(msg.UserID, msg.StorageID))
  132. if err != nil {
  133. logger.WithField("StorageID", msg.StorageID).
  134. Warnf("getting storage info: %s", err.Error())
  135. return nil, mq.Failed(errorcode.OperationFailed, "get storage info failed")
  136. }
  137. fullPath := filepath.Clean(filepath.Join(config.Cfg().StorageBaseDir, getStgResp.Directory, msg.Path))
  138. var uploadFilePathes []string
  139. err = filepath.WalkDir(fullPath, func(fname string, fi os.DirEntry, err error) error {
  140. if err != nil {
  141. return nil
  142. }
  143. if !fi.IsDir() {
  144. uploadFilePathes = append(uploadFilePathes, fname)
  145. }
  146. return nil
  147. })
  148. if err != nil {
  149. logger.Warnf("opening directory %s: %s", fullPath, err.Error())
  150. return nil, mq.Failed(errorcode.OperationFailed, "read directory failed")
  151. }
  152. objIter := iterator.NewUploadingObjectIterator(fullPath, uploadFilePathes)
  153. if msg.Redundancy.IsRepInfo() {
  154. repInfo, err := msg.Redundancy.ToRepInfo()
  155. if err != nil {
  156. logger.Warnf("getting rep redundancy info: %s", err.Error())
  157. return nil, mq.Failed(errorcode.OperationFailed, "get rep redundancy info failed")
  158. }
  159. tsk := svc.taskManager.StartNew(mytask.NewCreateRepPackage(msg.UserID, msg.BucketID, msg.Name, objIter, repInfo, msg.NodeAffinity))
  160. return mq.ReplyOK(agtmq.NewStartStorageCreatePackageResp(tsk.ID()))
  161. }
  162. ecInfo, err := msg.Redundancy.ToECInfo()
  163. if err != nil {
  164. logger.Warnf("getting ec redundancy info: %s", err.Error())
  165. return nil, mq.Failed(errorcode.OperationFailed, "get ec redundancy info failed")
  166. }
  167. tsk := svc.taskManager.StartNew(mytask.NewCreateECPackage(msg.UserID, msg.BucketID, msg.Name, objIter, ecInfo, msg.NodeAffinity))
  168. return mq.ReplyOK(agtmq.NewStartStorageCreatePackageResp(tsk.ID()))
  169. }
  170. func (svc *Service) WaitStorageCreatePackage(msg *agtmq.WaitStorageCreatePackage) (*agtmq.WaitStorageCreatePackageResp, *mq.CodeMessage) {
  171. tsk := svc.taskManager.FindByID(msg.TaskID)
  172. if tsk == nil {
  173. return nil, mq.Failed(errorcode.TaskNotFound, "task not found")
  174. }
  175. if msg.WaitTimeoutMs == 0 {
  176. tsk.Wait()
  177. } else if !tsk.WaitTimeout(time.Duration(msg.WaitTimeoutMs)) {
  178. return mq.ReplyOK(agtmq.NewWaitStorageCreatePackageResp(false, "", 0))
  179. }
  180. if tsk.Error() != nil {
  181. return mq.ReplyOK(agtmq.NewWaitStorageCreatePackageResp(true, tsk.Error().Error(), 0))
  182. }
  183. // TODO 避免判断类型
  184. if repTask, ok := tsk.Body().(*mytask.CreateRepPackage); ok {
  185. return mq.ReplyOK(agtmq.NewWaitStorageCreatePackageResp(true, "", repTask.Result.PackageID))
  186. }
  187. if ecTask, ok := tsk.Body().(*mytask.CreateECPackage); ok {
  188. return mq.ReplyOK(agtmq.NewWaitStorageCreatePackageResp(true, "", ecTask.Result.PackageID))
  189. }
  190. return nil, mq.Failed(errorcode.TaskNotFound, "task not found")
  191. }

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