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.5 kB

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

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