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

2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. package services
  2. import (
  3. "fmt"
  4. "time"
  5. stgsdk "gitlink.org.cn/cloudream/common/sdks/storage"
  6. mytask "gitlink.org.cn/cloudream/storage/client/internal/task"
  7. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  8. agtcmd "gitlink.org.cn/cloudream/storage/common/pkgs/cmd"
  9. "gitlink.org.cn/cloudream/storage/common/pkgs/db/model"
  10. "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
  11. "gitlink.org.cn/cloudream/storage/common/pkgs/iterator"
  12. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  13. )
  14. type PackageService struct {
  15. *Service
  16. }
  17. func (svc *Service) PackageSvc() *PackageService {
  18. return &PackageService{Service: svc}
  19. }
  20. func (svc *PackageService) Get(userID int64, packageID int64) (*model.Package, error) {
  21. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  22. if err != nil {
  23. return nil, fmt.Errorf("new coordinator client: %w", err)
  24. }
  25. defer stgglb.CoordinatorMQPool.Release(coorCli)
  26. getResp, err := coorCli.GetPackage(coormq.NewGetPackage(userID, packageID))
  27. if err != nil {
  28. return nil, fmt.Errorf("requsting to coodinator: %w", err)
  29. }
  30. return &getResp.Package, nil
  31. }
  32. func (svc *PackageService) DownloadPackage(userID int64, packageID int64) (iterator.DownloadingObjectIterator, error) {
  33. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  34. if err != nil {
  35. return nil, fmt.Errorf("new coordinator client: %w", err)
  36. }
  37. defer stgglb.CoordinatorMQPool.Release(coorCli)
  38. mutex, err := reqbuilder.NewBuilder().
  39. // 用于判断用户是否有对象权限
  40. Metadata().UserBucket().ReadAny().
  41. // 用于查询可用的下载节点
  42. Node().ReadAny().
  43. // 用于读取包信息
  44. Package().ReadOne(packageID).
  45. // 用于读取包内的文件信息
  46. Object().ReadAny().
  47. // 用于查询Rep配置
  48. ObjectRep().ReadAny().
  49. // 用于查询Block配置
  50. ObjectBlock().ReadAny().
  51. // 用于查询包含了副本的节点
  52. Cache().ReadAny().
  53. MutexLock(svc.DistLock)
  54. if err != nil {
  55. return nil, fmt.Errorf("acquire locks failed, err: %w", err)
  56. }
  57. getPkgResp, err := coorCli.GetPackage(coormq.NewGetPackage(userID, packageID))
  58. if err != nil {
  59. return nil, fmt.Errorf("getting package: %w", err)
  60. }
  61. getObjsResp, err := coorCli.GetPackageObjects(coormq.NewGetPackageObjects(userID, packageID))
  62. if err != nil {
  63. return nil, fmt.Errorf("getting package objects: %w", err)
  64. }
  65. if getPkgResp.Redundancy.IsRepInfo() {
  66. iter, err := svc.downloadRepPackage(packageID, getObjsResp.Objects, coorCli)
  67. if err != nil {
  68. mutex.Unlock()
  69. return nil, err
  70. }
  71. iter.OnClosing = func() {
  72. mutex.Unlock()
  73. }
  74. return iter, nil
  75. } else {
  76. iter, err := svc.downloadECPackage(getPkgResp.Package, getObjsResp.Objects, coorCli)
  77. if err != nil {
  78. mutex.Unlock()
  79. return nil, err
  80. }
  81. iter.OnClosing = func() {
  82. mutex.Unlock()
  83. }
  84. return iter, nil
  85. }
  86. }
  87. func (svc *PackageService) downloadRepPackage(packageID int64, objects []model.Object, coorCli *coormq.Client) (*iterator.RepObjectIterator, error) {
  88. getObjRepDataResp, err := coorCli.GetPackageObjectRepData(coormq.NewGetPackageObjectRepData(packageID))
  89. if err != nil {
  90. return nil, fmt.Errorf("getting package object rep data: %w", err)
  91. }
  92. iter := iterator.NewRepObjectIterator(objects, getObjRepDataResp.Data, &iterator.DownloadContext{
  93. Distlock: svc.DistLock,
  94. })
  95. return iter, nil
  96. }
  97. func (svc *PackageService) downloadECPackage(pkg model.Package, objects []model.Object, coorCli *coormq.Client) (*iterator.ECObjectIterator, error) {
  98. getObjECDataResp, err := coorCli.GetPackageObjectECData(coormq.NewGetPackageObjectECData(pkg.PackageID))
  99. if err != nil {
  100. return nil, fmt.Errorf("getting package object ec data: %w", err)
  101. }
  102. var ecInfo stgsdk.ECRedundancyInfo
  103. if ecInfo, err = pkg.Redundancy.ToECInfo(); err != nil {
  104. return nil, fmt.Errorf("get ec redundancy info: %w", err)
  105. }
  106. getECResp, err := coorCli.GetECConfig(coormq.NewGetECConfig(ecInfo.ECName))
  107. if err != nil {
  108. return nil, fmt.Errorf("getting ec: %w", err)
  109. }
  110. iter := iterator.NewECObjectIterator(objects, getObjECDataResp.Data, ecInfo, getECResp.Config, &iterator.DownloadContext{
  111. Distlock: svc.DistLock,
  112. })
  113. return iter, nil
  114. }
  115. func (svc *PackageService) StartCreatingRepPackage(userID int64, bucketID int64, name string, objIter iterator.UploadingObjectIterator, repInfo stgsdk.RepRedundancyInfo, nodeAffinity *int64) (string, error) {
  116. tsk := svc.TaskMgr.StartNew(mytask.NewCreateRepPackage(userID, bucketID, name, objIter, repInfo, nodeAffinity))
  117. return tsk.ID(), nil
  118. }
  119. func (svc *PackageService) WaitCreatingRepPackage(taskID string, waitTimeout time.Duration) (bool, *mytask.CreateRepPackageResult, error) {
  120. tsk := svc.TaskMgr.FindByID(taskID)
  121. if tsk.WaitTimeout(waitTimeout) {
  122. cteatePkgTask := tsk.Body().(*mytask.CreateRepPackage)
  123. return true, cteatePkgTask.Result, tsk.Error()
  124. }
  125. return false, nil, nil
  126. }
  127. func (svc *PackageService) StartUpdatingRepPackage(userID int64, packageID int64, objIter iterator.UploadingObjectIterator) (string, error) {
  128. tsk := svc.TaskMgr.StartNew(mytask.NewUpdateRepPackage(userID, packageID, objIter))
  129. return tsk.ID(), nil
  130. }
  131. func (svc *PackageService) WaitUpdatingRepPackage(taskID string, waitTimeout time.Duration) (bool, *agtcmd.UpdateRepPackageResult, error) {
  132. tsk := svc.TaskMgr.FindByID(taskID)
  133. if tsk.WaitTimeout(waitTimeout) {
  134. updatePkgTask := tsk.Body().(*mytask.UpdateRepPackage)
  135. return true, updatePkgTask.Result, tsk.Error()
  136. }
  137. return false, nil, nil
  138. }
  139. func (svc *PackageService) StartCreatingECPackage(userID int64, bucketID int64, name string, objIter iterator.UploadingObjectIterator, ecInfo stgsdk.ECRedundancyInfo, nodeAffinity *int64) (string, error) {
  140. tsk := svc.TaskMgr.StartNew(mytask.NewCreateECPackage(userID, bucketID, name, objIter, ecInfo, nodeAffinity))
  141. return tsk.ID(), nil
  142. }
  143. func (svc *PackageService) WaitCreatingECPackage(taskID string, waitTimeout time.Duration) (bool, *agtcmd.CreateRepPackageResult, error) {
  144. tsk := svc.TaskMgr.FindByID(taskID)
  145. if tsk.WaitTimeout(waitTimeout) {
  146. cteatePkgTask := tsk.Body().(*mytask.CreateRepPackage)
  147. return true, cteatePkgTask.Result, tsk.Error()
  148. }
  149. return false, nil, nil
  150. }
  151. func (svc *PackageService) StartUpdatingECPackage(userID int64, packageID int64, objIter iterator.UploadingObjectIterator) (string, error) {
  152. tsk := svc.TaskMgr.StartNew(mytask.NewUpdateECPackage(userID, packageID, objIter))
  153. return tsk.ID(), nil
  154. }
  155. func (svc *PackageService) WaitUpdatingECPackage(taskID string, waitTimeout time.Duration) (bool, *agtcmd.UpdateECPackageResult, error) {
  156. tsk := svc.TaskMgr.FindByID(taskID)
  157. if tsk.WaitTimeout(waitTimeout) {
  158. updatePkgTask := tsk.Body().(*mytask.UpdateECPackage)
  159. return true, updatePkgTask.Result, tsk.Error()
  160. }
  161. return false, nil, nil
  162. }
  163. func (svc *PackageService) DeletePackage(userID int64, packageID int64) error {
  164. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  165. if err != nil {
  166. return fmt.Errorf("new coordinator client: %w", err)
  167. }
  168. defer stgglb.CoordinatorMQPool.Release(coorCli)
  169. mutex, err := reqbuilder.NewBuilder().
  170. Metadata().
  171. // 用于判断用户是否有对象的权限
  172. UserBucket().ReadAny().
  173. // 用于读取、修改包信息
  174. Package().WriteOne(packageID).
  175. // 用于删除包内的所有文件
  176. Object().WriteAny().
  177. // 用于删除Rep配置
  178. ObjectRep().WriteAny().
  179. // 用于删除Block配置
  180. ObjectBlock().WriteAny().
  181. // 用于修改Move此Object的记录的状态
  182. StoragePackage().WriteAny().
  183. MutexLock(svc.DistLock)
  184. if err != nil {
  185. return fmt.Errorf("acquire locks failed, err: %w", err)
  186. }
  187. defer mutex.Unlock()
  188. _, err = coorCli.DeletePackage(coormq.NewDeletePackage(userID, packageID))
  189. if err != nil {
  190. return fmt.Errorf("deleting package: %w", err)
  191. }
  192. return nil
  193. }
  194. func (svc *PackageService) GetCachedNodes(userID int64, packageID int64) (stgsdk.PackageCachingInfo, error) {
  195. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  196. if err != nil {
  197. return stgsdk.PackageCachingInfo{}, fmt.Errorf("new coordinator client: %w", err)
  198. }
  199. defer stgglb.CoordinatorMQPool.Release(coorCli)
  200. resp, err := coorCli.GetPackageCachedNodes(coormq.NewGetPackageCachedNodes(userID, packageID))
  201. if err != nil {
  202. return stgsdk.PackageCachingInfo{}, fmt.Errorf("get package cached nodes: %w", err)
  203. }
  204. tmp := stgsdk.PackageCachingInfo{
  205. NodeInfos: resp.NodeInfos,
  206. PackageSize: resp.PackageSize,
  207. RedunancyType: resp.RedunancyType,
  208. }
  209. return tmp, nil
  210. }
  211. func (svc *PackageService) GetLoadedNodes(userID int64, packageID int64) ([]int64, error) {
  212. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  213. if err != nil {
  214. return nil, fmt.Errorf("new coordinator client: %w", err)
  215. }
  216. defer stgglb.CoordinatorMQPool.Release(coorCli)
  217. resp, err := coorCli.GetPackageLoadedNodes(coormq.NewGetPackageLoadedNodes(userID, packageID))
  218. if err != nil {
  219. return nil, fmt.Errorf("get package loaded nodes: %w", err)
  220. }
  221. return resp.NodeIDs, nil
  222. }

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