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.

bucket.go 3.4 kB

2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113
  1. package services
  2. import (
  3. "fmt"
  4. stgglb "gitlink.org.cn/cloudream/storage/common/globals"
  5. "gitlink.org.cn/cloudream/storage/common/pkgs/db/model"
  6. "gitlink.org.cn/cloudream/storage/common/pkgs/distlock/reqbuilder"
  7. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  8. )
  9. type BucketService struct {
  10. *Service
  11. }
  12. func (svc *Service) BucketSvc() *BucketService {
  13. return &BucketService{Service: svc}
  14. }
  15. func (svc *BucketService) GetBucket(userID int64, bucketID int64) (model.Bucket, error) {
  16. // TODO
  17. panic("not implement yet")
  18. }
  19. func (svc *BucketService) GetUserBuckets(userID int64) ([]model.Bucket, error) {
  20. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  21. if err != nil {
  22. return nil, fmt.Errorf("new coordinator client: %w", err)
  23. }
  24. defer coorCli.Close()
  25. resp, err := coorCli.GetUserBuckets(coormq.NewGetUserBuckets(userID))
  26. if err != nil {
  27. return nil, fmt.Errorf("get user buckets failed, err: %w", err)
  28. }
  29. return resp.Buckets, nil
  30. }
  31. func (svc *BucketService) GetBucketPackages(userID int64, bucketID int64) ([]model.Package, error) {
  32. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  33. if err != nil {
  34. return nil, fmt.Errorf("new coordinator client: %w", err)
  35. }
  36. defer coorCli.Close()
  37. resp, err := coorCli.GetBucketPackages(coormq.NewGetBucketPackages(userID, bucketID))
  38. if err != nil {
  39. return nil, fmt.Errorf("get bucket packages failed, err: %w", err)
  40. }
  41. return resp.Packages, nil
  42. }
  43. func (svc *BucketService) CreateBucket(userID int64, bucketName string) (int64, error) {
  44. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  45. if err != nil {
  46. return 0, fmt.Errorf("new coordinator client: %w", err)
  47. }
  48. defer coorCli.Close()
  49. // TODO 只有阅读了系统操作的源码,才能知道要加哪些锁,但用户的命令可能会调用不止一个系统操作。
  50. // 因此加锁的操作还是必须在用户命令里完成,但具体加锁的内容,则需要被封装起来与系统操作放到一起,方便管理,避免分散改动。
  51. mutex, err := reqbuilder.NewBuilder().
  52. Metadata().Bucket().CreateOne(userID, bucketName).
  53. // TODO 可以考虑二次加锁,加的更精确
  54. UserBucket().CreateAny().
  55. MutexLock(svc.DistLock)
  56. if err != nil {
  57. return 0, fmt.Errorf("acquire locks failed, err: %w", err)
  58. }
  59. defer mutex.Unlock()
  60. resp, err := coorCli.CreateBucket(coormq.NewCreateBucket(userID, bucketName))
  61. if err != nil {
  62. return 0, fmt.Errorf("creating bucket: %w", err)
  63. }
  64. return resp.BucketID, nil
  65. }
  66. func (svc *BucketService) DeleteBucket(userID int64, bucketID int64) error {
  67. coorCli, err := stgglb.CoordinatorMQPool.Acquire()
  68. if err != nil {
  69. return fmt.Errorf("new coordinator client: %w", err)
  70. }
  71. defer coorCli.Close()
  72. // TODO 检查用户是否有删除这个Bucket的权限。检查的时候可以只上UserBucket的Read锁
  73. mutex, err := reqbuilder.NewBuilder().
  74. Metadata().
  75. UserBucket().WriteAny().
  76. Bucket().WriteOne(bucketID).
  77. Package().WriteAny().
  78. Object().WriteAny().
  79. ObjectRep().WriteAny().
  80. ObjectBlock().WriteAny().
  81. StoragePackage().WriteAny().
  82. MutexLock(svc.DistLock)
  83. if err != nil {
  84. return fmt.Errorf("acquire locks failed, err: %w", err)
  85. }
  86. defer mutex.Unlock()
  87. _, err = coorCli.DeleteBucket(coormq.NewDeleteBucket(userID, bucketID))
  88. if err != nil {
  89. return fmt.Errorf("request to coordinator failed, err: %w", err)
  90. }
  91. return nil
  92. }

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