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

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. package db2
  2. import (
  3. "errors"
  4. "fmt"
  5. "gorm.io/gorm"
  6. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  7. "gitlink.org.cn/cloudream/storage/common/pkgs/db2/model"
  8. )
  9. type PackageDB struct {
  10. *DB
  11. }
  12. func (db *DB) Package() *PackageDB {
  13. return &PackageDB{DB: db}
  14. }
  15. func (db *PackageDB) GetByID(ctx SQLContext, packageID cdssdk.PackageID) (model.Package, error) {
  16. var ret model.Package
  17. err := ctx.Table("Package").Where("PackageID = ?", packageID).First(&ret).Error
  18. return ret, err
  19. }
  20. func (db *PackageDB) GetByName(ctx SQLContext, bucketID cdssdk.BucketID, name string) (model.Package, error) {
  21. var ret model.Package
  22. err := ctx.Table("Package").Where("BucketID = ? AND Name = ?", bucketID, name).First(&ret).Error
  23. return ret, err
  24. }
  25. func (db *PackageDB) BatchTestPackageID(ctx SQLContext, pkgIDs []cdssdk.PackageID) (map[cdssdk.PackageID]bool, error) {
  26. if len(pkgIDs) == 0 {
  27. return make(map[cdssdk.PackageID]bool), nil
  28. }
  29. var avaiIDs []cdssdk.PackageID
  30. err := ctx.Table("Package").
  31. Select("PackageID").
  32. Where("PackageID IN ?", pkgIDs).
  33. Find(&avaiIDs).Error
  34. if err != nil {
  35. return nil, err
  36. }
  37. avaiIDMap := make(map[cdssdk.PackageID]bool)
  38. for _, pkgID := range avaiIDs {
  39. avaiIDMap[pkgID] = true
  40. }
  41. return avaiIDMap, nil
  42. }
  43. func (*PackageDB) BatchGetAllPackageIDs(ctx SQLContext, start int, count int) ([]cdssdk.PackageID, error) {
  44. var ret []cdssdk.PackageID
  45. err := ctx.Table("Package").Select("PackageID").Limit(count).Offset(start).Find(&ret).Error
  46. return ret, err
  47. }
  48. func (db *PackageDB) GetBucketPackages(ctx SQLContext, userID cdssdk.UserID, bucketID cdssdk.BucketID) ([]model.Package, error) {
  49. var ret []model.Package
  50. err := ctx.Table("UserBucket").
  51. Select("Package.*").
  52. Joins("JOIN Package ON UserBucket.BucketID = Package.BucketID").
  53. Where("UserBucket.UserID = ? AND UserBucket.BucketID = ?", userID, bucketID).
  54. Find(&ret).Error
  55. return ret, err
  56. }
  57. // IsAvailable 判断一个用户是否拥有指定对象
  58. func (db *PackageDB) IsAvailable(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (bool, error) {
  59. var pkgID cdssdk.PackageID
  60. err := ctx.Table("Package").
  61. Select("Package.PackageID").
  62. Joins("JOIN UserBucket ON Package.BucketID = UserBucket.BucketID").
  63. Where("Package.PackageID = ? AND UserBucket.UserID = ?", packageID, userID).
  64. Scan(&pkgID).Error
  65. if err == gorm.ErrRecordNotFound {
  66. return false, nil
  67. }
  68. if err != nil {
  69. return false, fmt.Errorf("find package failed, err: %w", err)
  70. }
  71. return true, nil
  72. }
  73. // GetUserPackage 获得Package,如果用户没有权限访问,则不会获得结果
  74. func (db *PackageDB) GetUserPackage(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (model.Package, error) {
  75. var ret model.Package
  76. err := ctx.Table("Package").
  77. Select("Package.*").
  78. Joins("JOIN UserBucket ON Package.BucketID = UserBucket.BucketID").
  79. Where("Package.PackageID = ? AND UserBucket.UserID = ?", packageID, userID).
  80. First(&ret).Error
  81. return ret, err
  82. }
  83. // 在指定名称的Bucket中查找指定名称的Package
  84. func (*PackageDB) GetUserPackageByName(ctx SQLContext, userID cdssdk.UserID, bucketName string, packageName string) (model.Package, error) {
  85. var ret model.Package
  86. err := ctx.Table("Package").
  87. Select("Package.*").
  88. Joins("JOIN Bucket ON Package.BucketID = Bucket.BucketID").
  89. Joins("JOIN UserBucket ON Bucket.BucketID = UserBucket.BucketID").
  90. Where("Package.Name = ? AND Bucket.Name = ? AND UserBucket.UserID = ?", packageName, bucketName, userID).
  91. First(&ret).Error
  92. return ret, err
  93. }
  94. func (db *PackageDB) Create(ctx SQLContext, bucketID cdssdk.BucketID, name string) (cdssdk.PackageID, error) {
  95. var packageID int64
  96. err := ctx.Table("Package").
  97. Select("PackageID").
  98. Where("Name = ? AND BucketID = ?", name, bucketID).
  99. Scan(&packageID).Error
  100. if err != nil {
  101. return 0, err
  102. }
  103. if packageID != 0 {
  104. return 0, errors.New("package already exists")
  105. }
  106. newPackage := cdssdk.Package{Name: name, BucketID: bucketID, State: cdssdk.PackageStateNormal}
  107. if err := ctx.Create(&newPackage).Error; err != nil {
  108. return 0, fmt.Errorf("insert package failed, err: %w", err)
  109. }
  110. return newPackage.PackageID, nil
  111. }
  112. // SoftDelete 设置一个对象被删除,并将相关数据删除
  113. func (db *PackageDB) SoftDelete(ctx SQLContext, packageID cdssdk.PackageID) error {
  114. obj, err := db.GetByID(ctx, packageID)
  115. if err != nil {
  116. return fmt.Errorf("get package failed, err: %w", err)
  117. }
  118. if obj.State != cdssdk.PackageStateNormal {
  119. return nil
  120. }
  121. if err := db.ChangeState(ctx, packageID, cdssdk.PackageStateDeleted); err != nil {
  122. return fmt.Errorf("change package state failed, err: %w", err)
  123. }
  124. if err := db.ObjectAccessStat().DeleteInPackage(ctx, packageID); err != nil {
  125. return fmt.Errorf("delete from object access stat: %w", err)
  126. }
  127. if err := db.ObjectBlock().DeleteInPackage(ctx, packageID); err != nil {
  128. return fmt.Errorf("delete from object block failed, err: %w", err)
  129. }
  130. if err := db.PinnedObject().DeleteInPackage(ctx, packageID); err != nil {
  131. return fmt.Errorf("deleting pinned objects in package: %w", err)
  132. }
  133. if err := db.Object().DeleteInPackage(ctx, packageID); err != nil {
  134. return fmt.Errorf("deleting objects in package: %w", err)
  135. }
  136. if _, err := db.StoragePackage().SetAllPackageDeleted(ctx, packageID); err != nil {
  137. return fmt.Errorf("set storage package deleted failed, err: %w", err)
  138. }
  139. return nil
  140. }
  141. // DeleteUnused 删除一个已经是Deleted状态,且不再被使用的对象
  142. func (PackageDB) DeleteUnused(ctx SQLContext, packageID cdssdk.PackageID) error {
  143. err := ctx.Exec("DELETE FROM Package WHERE PackageID = ? AND State = ? AND NOT EXISTS (SELECT StorageID FROM StoragePackage WHERE PackageID = ?)",
  144. packageID,
  145. cdssdk.PackageStateDeleted,
  146. packageID,
  147. ).Error
  148. return err
  149. }
  150. func (*PackageDB) ChangeState(ctx SQLContext, packageID cdssdk.PackageID, state string) error {
  151. err := ctx.Exec("UPDATE Package SET State = ? WHERE PackageID = ?", state, packageID).Error
  152. return err
  153. }

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