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

2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package db
  2. import (
  3. "database/sql"
  4. "errors"
  5. "fmt"
  6. "github.com/jmoiron/sqlx"
  7. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  8. "gitlink.org.cn/cloudream/storage/common/pkgs/db/model"
  9. )
  10. type PackageDB struct {
  11. *DB
  12. }
  13. func (db *DB) Package() *PackageDB {
  14. return &PackageDB{DB: db}
  15. }
  16. func (db *PackageDB) GetByID(ctx SQLContext, packageID cdssdk.PackageID) (model.Package, error) {
  17. var ret model.Package
  18. err := sqlx.Get(ctx, &ret, "select * from Package where PackageID = ?", packageID)
  19. return ret, err
  20. }
  21. func (db *PackageDB) GetByName(ctx SQLContext, bucketID cdssdk.BucketID, name string) (model.Package, error) {
  22. var ret model.Package
  23. err := sqlx.Get(ctx, &ret, "select * from Package where BucketID = ? and Name = ?", bucketID, name)
  24. return ret, err
  25. }
  26. func (*PackageDB) BatchGetAllPackageIDs(ctx SQLContext, start int, count int) ([]cdssdk.PackageID, error) {
  27. var ret []cdssdk.PackageID
  28. err := sqlx.Select(ctx, &ret, "select PackageID from Package limit ?, ?", start, count)
  29. return ret, err
  30. }
  31. func (db *PackageDB) GetBucketPackages(ctx SQLContext, userID cdssdk.UserID, bucketID cdssdk.BucketID) ([]model.Package, error) {
  32. var ret []model.Package
  33. err := sqlx.Select(ctx, &ret, "select Package.* from UserBucket, Package where UserID = ? and UserBucket.BucketID = ? and UserBucket.BucketID = Package.BucketID", userID, bucketID)
  34. return ret, err
  35. }
  36. // IsAvailable 判断一个用户是否拥有指定对象
  37. func (db *PackageDB) IsAvailable(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (bool, error) {
  38. var pkgID cdssdk.PackageID
  39. // 先根据PackageID找到Package,然后判断此Package所在的Bucket是不是归此用户所有
  40. err := sqlx.Get(ctx, &pkgID,
  41. "select Package.PackageID from Package, UserBucket where "+
  42. "Package.PackageID = ? and "+
  43. "Package.BucketID = UserBucket.BucketID and "+
  44. "UserBucket.UserID = ?",
  45. packageID, userID)
  46. if err == sql.ErrNoRows {
  47. return false, nil
  48. }
  49. if err != nil {
  50. return false, fmt.Errorf("find package failed, err: %w", err)
  51. }
  52. return true, nil
  53. }
  54. // GetUserPackage 获得Package,如果用户没有权限访问,则不会获得结果
  55. func (db *PackageDB) GetUserPackage(ctx SQLContext, userID cdssdk.UserID, packageID cdssdk.PackageID) (model.Package, error) {
  56. var ret model.Package
  57. err := sqlx.Get(ctx, &ret,
  58. "select Package.* from Package, UserBucket where"+
  59. " Package.PackageID = ? and"+
  60. " Package.BucketID = UserBucket.BucketID and"+
  61. " UserBucket.UserID = ?",
  62. packageID, userID)
  63. return ret, err
  64. }
  65. func (db *PackageDB) Create(ctx SQLContext, bucketID cdssdk.BucketID, name string) (cdssdk.PackageID, error) {
  66. // 根据packagename和bucketid查询,若不存在则插入,若存在则返回错误
  67. var packageID int64
  68. err := sqlx.Get(ctx, &packageID, "select PackageID from Package where Name = ? AND BucketID = ?", name, bucketID)
  69. // 无错误代表存在记录
  70. if err == nil {
  71. return 0, fmt.Errorf("package with given Name and BucketID already exists")
  72. }
  73. // 错误不是记录不存在
  74. if !errors.Is(err, sql.ErrNoRows) {
  75. return 0, fmt.Errorf("query Package by PackageName and BucketID failed, err: %w", err)
  76. }
  77. sql := "insert into Package(Name, BucketID, State) values(?,?,?)"
  78. r, err := ctx.Exec(sql, name, bucketID, cdssdk.PackageStateNormal)
  79. if err != nil {
  80. return 0, fmt.Errorf("insert package failed, err: %w", err)
  81. }
  82. packageID, err = r.LastInsertId()
  83. if err != nil {
  84. return 0, fmt.Errorf("get id of inserted package failed, err: %w", err)
  85. }
  86. return cdssdk.PackageID(packageID), nil
  87. }
  88. // SoftDelete 设置一个对象被删除,并将相关数据删除
  89. func (db *PackageDB) SoftDelete(ctx SQLContext, packageID cdssdk.PackageID) error {
  90. obj, err := db.GetByID(ctx, packageID)
  91. if err != nil {
  92. return fmt.Errorf("get package failed, err: %w", err)
  93. }
  94. // 不是正常状态的Package,则不删除
  95. // TODO 未来可能有其他状态
  96. if obj.State != cdssdk.PackageStateNormal {
  97. return nil
  98. }
  99. err = db.ChangeState(ctx, packageID, cdssdk.PackageStateDeleted)
  100. if err != nil {
  101. return fmt.Errorf("change package state failed, err: %w", err)
  102. }
  103. err = db.ObjectBlock().DeleteInPackage(ctx, packageID)
  104. if err != nil {
  105. return fmt.Errorf("delete from object rep failed, err: %w", err)
  106. }
  107. if err := db.Object().DeleteInPackage(ctx, packageID); err != nil {
  108. return fmt.Errorf("deleting objects in package: %w", err)
  109. }
  110. _, err = db.StoragePackage().SetAllPackageDeleted(ctx, packageID)
  111. if err != nil {
  112. return fmt.Errorf("set storage package deleted failed, err: %w", err)
  113. }
  114. return nil
  115. }
  116. // DeleteUnused 删除一个已经是Deleted状态,且不再被使用的对象。目前可能被使用的地方只有StoragePackage
  117. func (PackageDB) DeleteUnused(ctx SQLContext, packageID cdssdk.PackageID) error {
  118. _, err := ctx.Exec("delete from Package where PackageID = ? and State = ? and "+
  119. "not exists(select StorageID from StoragePackage where PackageID = ?)",
  120. packageID,
  121. cdssdk.PackageStateDeleted,
  122. packageID,
  123. )
  124. return err
  125. }
  126. func (*PackageDB) ChangeState(ctx SQLContext, packageID cdssdk.PackageID, state string) error {
  127. _, err := ctx.Exec("update Package set State = ? where PackageID = ?", state, packageID)
  128. return err
  129. }

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