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

2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. package db
  2. import (
  3. "database/sql"
  4. "errors"
  5. "fmt"
  6. "github.com/jmoiron/sqlx"
  7. "github.com/samber/lo"
  8. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  9. "gitlink.org.cn/cloudream/storage/common/pkgs/db/model"
  10. )
  11. type PackageDB struct {
  12. *DB
  13. }
  14. func (db *DB) Package() *PackageDB {
  15. return &PackageDB{DB: db}
  16. }
  17. func (db *PackageDB) GetByID(ctx SQLContext, packageID cdssdk.PackageID) (model.Package, error) {
  18. var ret model.Package
  19. err := sqlx.Get(ctx, &ret, "select * from Package where PackageID = ?", packageID)
  20. return ret, err
  21. }
  22. func (db *PackageDB) GetByName(ctx SQLContext, bucketID cdssdk.BucketID, name string) (model.Package, error) {
  23. var ret model.Package
  24. err := sqlx.Get(ctx, &ret, "select * from Package where BucketID = ? and Name = ?", bucketID, name)
  25. return ret, err
  26. }
  27. func (db *PackageDB) BatchTestPackageID(ctx SQLContext, pkgIDs []cdssdk.PackageID) (map[cdssdk.PackageID]bool, error) {
  28. stmt, args, err := sqlx.In("select PackageID from Package where PackageID in (?)", lo.Uniq(pkgIDs))
  29. if err != nil {
  30. return nil, err
  31. }
  32. var avaiIDs []cdssdk.PackageID
  33. err = sqlx.Select(ctx, &avaiIDs, stmt, args...)
  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 := sqlx.Select(ctx, &ret, "select PackageID from Package limit ?, ?", start, count)
  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 := sqlx.Select(ctx, &ret, "select Package.* from UserBucket, Package where UserID = ? and UserBucket.BucketID = ? and UserBucket.BucketID = Package.BucketID", userID, bucketID)
  51. return ret, err
  52. }
  53. // IsAvailable 判断一个用户是否拥有指定对象
  54. func (db *PackageDB) IsAvailable(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (bool, error) {
  55. var pkgID cdssdk.PackageID
  56. // 先根据PackageID找到Package,然后判断此Package所在的Bucket是不是归此用户所有
  57. err := sqlx.Get(ctx, &pkgID,
  58. "select Package.PackageID from Package, UserBucket where "+
  59. "Package.PackageID = ? and "+
  60. "Package.BucketID = UserBucket.BucketID and "+
  61. "UserBucket.UserID = ?",
  62. packageID, userID)
  63. if err == sql.ErrNoRows {
  64. return false, nil
  65. }
  66. if err != nil {
  67. return false, fmt.Errorf("find package failed, err: %w", err)
  68. }
  69. return true, nil
  70. }
  71. // GetUserPackage 获得Package,如果用户没有权限访问,则不会获得结果
  72. func (db *PackageDB) GetUserPackage(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (model.Package, error) {
  73. var ret model.Package
  74. err := sqlx.Get(ctx, &ret,
  75. "select Package.* from Package, UserBucket where"+
  76. " Package.PackageID = ? and"+
  77. " Package.BucketID = UserBucket.BucketID and"+
  78. " UserBucket.UserID = ?",
  79. packageID, userID)
  80. return ret, err
  81. }
  82. // 在指定名称的Bucket中查找指定名称的Package
  83. func (*PackageDB) GetUserPackageByName(ctx SQLContext, userID cdssdk.UserID, bucketName string, packageName string) (cdssdk.Package, error) {
  84. var ret model.Package
  85. err := sqlx.Get(ctx, &ret,
  86. "select Package.* from Package, Bucket, UserBucket where"+
  87. " Package.Name = ? and"+
  88. " Package.BucketID = Bucket.BucketID and"+
  89. " Bucket.Name = ? and"+
  90. " UserBucket.UserID = ? and"+
  91. " UserBucket.BucketID = Bucket.BucketID",
  92. packageName, bucketName, userID)
  93. return ret, err
  94. }
  95. func (db *PackageDB) Create(ctx SQLContext, bucketID cdssdk.BucketID, name string) (cdssdk.PackageID, error) {
  96. // 根据packagename和bucketid查询,若不存在则插入,若存在则返回错误
  97. var packageID int64
  98. err := sqlx.Get(ctx, &packageID, "select PackageID from Package where Name = ? AND BucketID = ? for update", name, bucketID)
  99. // 无错误代表存在记录
  100. if err == nil {
  101. return 0, fmt.Errorf("package with given Name and BucketID already exists")
  102. }
  103. // 错误不是记录不存在
  104. if !errors.Is(err, sql.ErrNoRows) {
  105. return 0, fmt.Errorf("query Package by PackageName and BucketID failed, err: %w", err)
  106. }
  107. sql := "insert into Package(Name, BucketID, State) values(?,?,?)"
  108. r, err := ctx.Exec(sql, name, bucketID, cdssdk.PackageStateNormal)
  109. if err != nil {
  110. return 0, fmt.Errorf("insert package failed, err: %w", err)
  111. }
  112. packageID, err = r.LastInsertId()
  113. if err != nil {
  114. return 0, fmt.Errorf("get id of inserted package failed, err: %w", err)
  115. }
  116. return cdssdk.PackageID(packageID), nil
  117. }
  118. // SoftDelete 设置一个对象被删除,并将相关数据删除
  119. func (db *PackageDB) SoftDelete(ctx SQLContext, packageID cdssdk.PackageID) error {
  120. obj, err := db.GetByID(ctx, packageID)
  121. if err != nil {
  122. return fmt.Errorf("get package failed, err: %w", err)
  123. }
  124. // 不是正常状态的Package,则不删除
  125. // TODO 未来可能有其他状态
  126. if obj.State != cdssdk.PackageStateNormal {
  127. return nil
  128. }
  129. err = db.ChangeState(ctx, packageID, cdssdk.PackageStateDeleted)
  130. if err != nil {
  131. return fmt.Errorf("change package state failed, err: %w", err)
  132. }
  133. err = db.ObjectAccessStat().DeleteInPackage(ctx, packageID)
  134. if err != nil {
  135. return fmt.Errorf("delete from object access stat: %w", err)
  136. }
  137. err = db.ObjectBlock().DeleteInPackage(ctx, packageID)
  138. if err != nil {
  139. return fmt.Errorf("delete from object rep failed, err: %w", err)
  140. }
  141. if err := db.PinnedObject().DeleteInPackage(ctx, packageID); err != nil {
  142. return fmt.Errorf("deleting pinned objects in package: %w", err)
  143. }
  144. if err := db.Object().DeleteInPackage(ctx, packageID); err != nil {
  145. return fmt.Errorf("deleting objects in package: %w", err)
  146. }
  147. _, err = db.StoragePackage().SetAllPackageDeleted(ctx, packageID)
  148. if err != nil {
  149. return fmt.Errorf("set storage package deleted failed, err: %w", err)
  150. }
  151. return nil
  152. }
  153. // DeleteUnused 删除一个已经是Deleted状态,且不再被使用的对象。目前可能被使用的地方只有StoragePackage
  154. func (PackageDB) DeleteUnused(ctx SQLContext, packageID cdssdk.PackageID) error {
  155. _, err := ctx.Exec("delete from Package where PackageID = ? and State = ? and "+
  156. "not exists(select StorageID from StoragePackage where PackageID = ?)",
  157. packageID,
  158. cdssdk.PackageStateDeleted,
  159. packageID,
  160. )
  161. return err
  162. }
  163. func (*PackageDB) ChangeState(ctx SQLContext, packageID cdssdk.PackageID, state string) error {
  164. _, err := ctx.Exec("update Package set State = ? where PackageID = ?", state, packageID)
  165. return err
  166. }

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