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.

object.go 9.7 kB

2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  1. package mq
  2. import (
  3. "database/sql"
  4. "fmt"
  5. "github.com/jmoiron/sqlx"
  6. "github.com/samber/lo"
  7. "gitlink.org.cn/cloudream/common/consts/errorcode"
  8. "gitlink.org.cn/cloudream/common/pkgs/logger"
  9. "gitlink.org.cn/cloudream/common/pkgs/mq"
  10. cdssdk "gitlink.org.cn/cloudream/common/sdks/storage"
  11. "gitlink.org.cn/cloudream/common/utils/sort2"
  12. stgmod "gitlink.org.cn/cloudream/storage/common/models"
  13. coormq "gitlink.org.cn/cloudream/storage/common/pkgs/mq/coordinator"
  14. )
  15. func (svc *Service) GetPackageObjects(msg *coormq.GetPackageObjects) (*coormq.GetPackageObjectsResp, *mq.CodeMessage) {
  16. var objs []cdssdk.Object
  17. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  18. _, err := svc.db.Package().GetUserPackage(tx, msg.UserID, msg.PackageID)
  19. if err != nil {
  20. return fmt.Errorf("getting package by id: %w", err)
  21. }
  22. objs, err = svc.db.Object().GetPackageObjects(svc.db.SQLCtx(), msg.PackageID)
  23. if err != nil {
  24. return fmt.Errorf("getting package objects: %w", err)
  25. }
  26. return nil
  27. })
  28. if err != nil {
  29. logger.WithField("UserID", msg.UserID).WithField("PackageID", msg.PackageID).
  30. Warn(err.Error())
  31. return nil, mq.Failed(errorcode.OperationFailed, "get package objects failed")
  32. }
  33. return mq.ReplyOK(coormq.NewGetPackageObjectsResp(objs))
  34. }
  35. func (svc *Service) GetPackageObjectDetails(msg *coormq.GetPackageObjectDetails) (*coormq.GetPackageObjectDetailsResp, *mq.CodeMessage) {
  36. var details []stgmod.ObjectDetail
  37. // 必须放在事务里进行,因为GetPackageBlockDetails是由多次数据库操作组成,必须保证数据的一致性
  38. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  39. var err error
  40. _, err = svc.db.Package().GetByID(tx, msg.PackageID)
  41. if err != nil {
  42. return fmt.Errorf("getting package by id: %w", err)
  43. }
  44. details, err = svc.db.Object().GetPackageObjectDetails(tx, msg.PackageID)
  45. if err != nil {
  46. return fmt.Errorf("getting package block details: %w", err)
  47. }
  48. return nil
  49. })
  50. if err != nil {
  51. logger.WithField("PackageID", msg.PackageID).Warn(err.Error())
  52. return nil, mq.Failed(errorcode.OperationFailed, "get package object block details failed")
  53. }
  54. return mq.ReplyOK(coormq.NewGetPackageObjectDetailsResp(details))
  55. }
  56. func (svc *Service) GetObjectDetails(msg *coormq.GetObjectDetails) (*coormq.GetObjectDetailsResp, *mq.CodeMessage) {
  57. details := make([]*stgmod.ObjectDetail, len(msg.ObjectIDs))
  58. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  59. var err error
  60. msg.ObjectIDs = sort2.SortAsc(msg.ObjectIDs)
  61. // 根据ID依次查询Object,ObjectBlock,PinnedObject,并根据升序的特点进行合并
  62. objs, err := svc.db.Object().BatchGet(tx, msg.ObjectIDs)
  63. if err != nil {
  64. return fmt.Errorf("batch get objects: %w", err)
  65. }
  66. objIDIdx := 0
  67. objIdx := 0
  68. for objIDIdx < len(msg.ObjectIDs) && objIdx < len(objs) {
  69. if msg.ObjectIDs[objIDIdx] < objs[objIdx].ObjectID {
  70. objIDIdx++
  71. continue
  72. }
  73. // 由于是使用msg.ObjectIDs去查询Object,因此不存在msg.ObjectIDs > Object.ObjectID的情况,
  74. // 下面同理
  75. obj := stgmod.ObjectDetail{
  76. Object: objs[objIDIdx],
  77. }
  78. details[objIDIdx] = &obj
  79. objIdx++
  80. }
  81. // 查询合并
  82. blocks, err := svc.db.ObjectBlock().BatchGetByObjectID(tx, msg.ObjectIDs)
  83. if err != nil {
  84. return fmt.Errorf("batch get object blocks: %w", err)
  85. }
  86. objIDIdx = 0
  87. blkIdx := 0
  88. for objIDIdx < len(msg.ObjectIDs) && blkIdx < len(blocks) {
  89. if details[objIDIdx] == nil {
  90. objIDIdx++
  91. continue
  92. }
  93. if msg.ObjectIDs[objIDIdx] < blocks[blkIdx].ObjectID {
  94. objIDIdx++
  95. continue
  96. }
  97. details[objIDIdx].Blocks = append(details[objIDIdx].Blocks, blocks[blkIdx])
  98. blkIdx++
  99. }
  100. // 查询合并
  101. pinneds, err := svc.db.PinnedObject().BatchGetByObjectID(tx, msg.ObjectIDs)
  102. if err != nil {
  103. return fmt.Errorf("batch get pinned objects: %w", err)
  104. }
  105. objIDIdx = 0
  106. pinIdx := 0
  107. for objIDIdx < len(msg.ObjectIDs) && pinIdx < len(pinneds) {
  108. if details[objIDIdx] == nil {
  109. objIDIdx++
  110. continue
  111. }
  112. if msg.ObjectIDs[objIDIdx] < pinneds[pinIdx].ObjectID {
  113. objIDIdx++
  114. continue
  115. }
  116. details[objIDIdx].PinnedAt = append(details[objIDIdx].PinnedAt, pinneds[pinIdx].NodeID)
  117. pinIdx++
  118. }
  119. return nil
  120. })
  121. if err != nil {
  122. logger.Warn(err.Error())
  123. return nil, mq.Failed(errorcode.OperationFailed, "get object details failed")
  124. }
  125. return mq.ReplyOK(coormq.RespGetObjectDetails(details))
  126. }
  127. func (svc *Service) UpdateObjectRedundancy(msg *coormq.UpdateObjectRedundancy) (*coormq.UpdateObjectRedundancyResp, *mq.CodeMessage) {
  128. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  129. return svc.db.Object().BatchUpdateRedundancy(tx, msg.Updatings)
  130. })
  131. if err != nil {
  132. logger.Warnf("batch updating redundancy: %s", err.Error())
  133. return nil, mq.Failed(errorcode.OperationFailed, "batch update redundancy failed")
  134. }
  135. return mq.ReplyOK(coormq.RespUpdateObjectRedundancy())
  136. }
  137. func (svc *Service) UpdateObjectInfos(msg *coormq.UpdateObjectInfos) (*coormq.UpdateObjectInfosResp, *mq.CodeMessage) {
  138. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  139. msg.Updatings = sort2.Sort(msg.Updatings, func(o1, o2 cdssdk.UpdatingObject) int {
  140. return sort2.Cmp(o1.ObjectID, o2.ObjectID)
  141. })
  142. objIDs := make([]cdssdk.ObjectID, len(msg.Updatings))
  143. for i, obj := range msg.Updatings {
  144. objIDs[i] = obj.ObjectID
  145. }
  146. oldObjs, err := svc.db.Object().BatchGet(tx, objIDs)
  147. if err != nil {
  148. return fmt.Errorf("batch getting objects: %w", err)
  149. }
  150. oldObjIDs := make([]cdssdk.ObjectID, len(oldObjs))
  151. for i, obj := range oldObjs {
  152. oldObjIDs[i] = obj.ObjectID
  153. }
  154. avaiUpdatings, notExistsObjs := pickByObjectIDs(msg.Updatings, oldObjIDs)
  155. if len(notExistsObjs) > 0 {
  156. // TODO 部分对象已经不存在
  157. }
  158. // 筛选出PackageID变化、Path变化的对象,这两种对象要检测改变后是否有冲突
  159. // 否则,直接更新即可
  160. //var pkgIDChangedObjs []cdssdk.Object
  161. //var pathChangedObjs []cdssdk.Object
  162. //var infoChangedObjs []cdssdk.Object
  163. //for i := range willUpdateObjs {
  164. // if willUpdateObjs[i].PackageID != oldObjs[i].PackageID {
  165. // newObj := oldObjs[i]
  166. // willUpdateObjs[i].ApplyTo(&newObj)
  167. // pkgIDChangedObjs = append(pkgIDChangedObjs, newObj)
  168. // } else if willUpdateObjs[i].Path != oldObjs[i].Path {
  169. // newObj := oldObjs[i]
  170. // willUpdateObjs[i].ApplyTo(&newObj)
  171. // pathChangedObjs = append(pathChangedObjs, newObj)
  172. // } else {
  173. // newObj := oldObjs[i]
  174. // willUpdateObjs[i].ApplyTo(&newObj)
  175. // infoChangedObjs = append(infoChangedObjs, newObj)
  176. // }
  177. //}
  178. newObjs := make([]cdssdk.Object, len(avaiUpdatings))
  179. for i := range newObjs {
  180. newObjs[i] = oldObjs[i]
  181. avaiUpdatings[i].ApplyTo(&newObjs[i])
  182. }
  183. err = svc.db.Object().BatchCreateOrUpdate(tx, newObjs)
  184. if err != nil {
  185. return fmt.Errorf("batch create or update: %w", err)
  186. }
  187. return nil
  188. })
  189. if err != nil {
  190. logger.Warnf("batch updating objects: %s", err.Error())
  191. return nil, mq.Failed(errorcode.OperationFailed, "batch update objects failed")
  192. }
  193. return mq.ReplyOK(coormq.RespUpdateObjectInfos())
  194. }
  195. // 根据objIDs从objs中挑选Object。
  196. // len(objs) >= len(objIDs)
  197. func pickByObjectIDs(objs []cdssdk.UpdatingObject, objIDs []cdssdk.ObjectID) (pickedObjs []cdssdk.UpdatingObject, notFoundObjs []cdssdk.UpdatingObject) {
  198. objIdx := 0
  199. idIdx := 0
  200. for idIdx < len(objIDs) && objIdx < len(objs) {
  201. if objs[objIdx].ObjectID < objIDs[idIdx] {
  202. notFoundObjs = append(notFoundObjs, objs[objIdx])
  203. objIdx++
  204. continue
  205. }
  206. pickedObjs = append(pickedObjs, objs[objIdx])
  207. objIdx++
  208. idIdx++
  209. }
  210. return
  211. }
  212. func (svc *Service) ensurePackageChangedObjects(tx *sqlx.Tx, objs []cdssdk.Object) ([]cdssdk.Object, error) {
  213. type PackageObjects struct {
  214. PackageID cdssdk.PackageID
  215. ObjectByPath map[string]*cdssdk.Object
  216. }
  217. packages := make(map[cdssdk.PackageID]*PackageObjects)
  218. for _, obj := range objs {
  219. pkg, ok := packages[obj.PackageID]
  220. if !ok {
  221. pkg = &PackageObjects{
  222. PackageID: obj.PackageID,
  223. ObjectByPath: make(map[string]*cdssdk.Object),
  224. }
  225. packages[obj.PackageID] = pkg
  226. }
  227. if pkg.ObjectByPath[obj.Path] == nil {
  228. o := obj
  229. pkg.ObjectByPath[obj.Path] = &o
  230. } else {
  231. // TODO 有冲突
  232. }
  233. }
  234. var willUpdateObjs []cdssdk.Object
  235. for _, pkg := range packages {
  236. existsObjs, err := svc.db.Object().BatchByPackagePath(tx, pkg.PackageID, lo.Keys(pkg.ObjectByPath))
  237. if err != nil {
  238. return nil, fmt.Errorf("batch getting objects by package path: %w", err)
  239. }
  240. for _, obj := range existsObjs {
  241. pkg.ObjectByPath[obj.Path] = nil
  242. }
  243. for _, obj := range pkg.ObjectByPath {
  244. if obj == nil {
  245. continue
  246. }
  247. willUpdateObjs = append(willUpdateObjs, *obj)
  248. }
  249. }
  250. return willUpdateObjs, nil
  251. }
  252. func (svc *Service) DeleteObjects(msg *coormq.DeleteObjects) (*coormq.DeleteObjectsResp, *mq.CodeMessage) {
  253. err := svc.db.DoTx(sql.LevelSerializable, func(tx *sqlx.Tx) error {
  254. err := svc.db.Object().BatchDelete(tx, msg.ObjectIDs)
  255. if err != nil {
  256. return fmt.Errorf("batch deleting objects: %w", err)
  257. }
  258. err = svc.db.ObjectBlock().BatchDeleteByObjectID(tx, msg.ObjectIDs)
  259. if err != nil {
  260. return fmt.Errorf("batch deleting object blocks: %w", err)
  261. }
  262. err = svc.db.PinnedObject().BatchDeleteByObjectID(tx, msg.ObjectIDs)
  263. if err != nil {
  264. return fmt.Errorf("batch deleting pinned objects: %w", err)
  265. }
  266. return nil
  267. })
  268. if err != nil {
  269. logger.Warnf("batch deleting objects: %s", err.Error())
  270. return nil, mq.Failed(errorcode.OperationFailed, "batch delete objects failed")
  271. }
  272. return mq.ReplyOK(coormq.RespDeleteObjects())
  273. }

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