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.

ipfs_lock.go 3.3 kB

2 years ago
2 years ago
2 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. package lockprovider
  2. import (
  3. "fmt"
  4. "gitlink.org.cn/cloudream/common/pkgs/distlock"
  5. "gitlink.org.cn/cloudream/common/utils/lo2"
  6. )
  7. const (
  8. IPFSLockPathPrefix = "IPFS"
  9. IPFSNodeIDPathIndex = 1
  10. IPFSBuzyLock = "Buzy"
  11. IPFSGCLock = "GC"
  12. )
  13. type IPFSLock struct {
  14. nodeLocks map[string]*IPFSNodeLock
  15. dummyLock *IPFSNodeLock
  16. }
  17. func NewIPFSLock() *IPFSLock {
  18. return &IPFSLock{
  19. nodeLocks: make(map[string]*IPFSNodeLock),
  20. dummyLock: NewIPFSNodeLock(),
  21. }
  22. }
  23. // CanLock 判断这个锁能否锁定成功
  24. func (l *IPFSLock) CanLock(lock distlock.Lock) error {
  25. nodeLock, ok := l.nodeLocks[lock.Path[IPFSNodeIDPathIndex]]
  26. if !ok {
  27. // 不能直接返回nil,因为如果锁数据的格式不对,也不能获取锁。
  28. // 这里使用一个空Provider来进行检查。
  29. return l.dummyLock.CanLock(lock)
  30. }
  31. return nodeLock.CanLock(lock)
  32. }
  33. // 锁定。在内部可以不用判断能否加锁,外部需要保证调用此函数前调用了CanLock进行检查
  34. func (l *IPFSLock) Lock(reqID string, lock distlock.Lock) error {
  35. nodeID := lock.Path[IPFSNodeIDPathIndex]
  36. nodeLock, ok := l.nodeLocks[nodeID]
  37. if !ok {
  38. nodeLock = NewIPFSNodeLock()
  39. l.nodeLocks[nodeID] = nodeLock
  40. }
  41. return nodeLock.Lock(reqID, lock)
  42. }
  43. // 解锁
  44. func (l *IPFSLock) Unlock(reqID string, lock distlock.Lock) error {
  45. nodeID := lock.Path[IPFSNodeIDPathIndex]
  46. nodeLock, ok := l.nodeLocks[nodeID]
  47. if !ok {
  48. return nil
  49. }
  50. return nodeLock.Unlock(reqID, lock)
  51. }
  52. // GetTargetString 将锁对象序列化为字符串,方便存储到ETCD
  53. func (l *IPFSLock) GetTargetString(target any) (string, error) {
  54. tar := target.(StringLockTarget)
  55. return StringLockTargetToString(&tar)
  56. }
  57. // ParseTargetString 解析字符串格式的锁对象数据
  58. func (l *IPFSLock) ParseTargetString(targetStr string) (any, error) {
  59. return StringLockTargetFromString(targetStr)
  60. }
  61. // Clear 清除内部所有状态
  62. func (l *IPFSLock) Clear() {
  63. l.nodeLocks = make(map[string]*IPFSNodeLock)
  64. }
  65. type IPFSNodeLock struct {
  66. buzyReqIDs []string
  67. gcReqIDs []string
  68. lockCompatibilityTable *LockCompatibilityTable
  69. }
  70. func NewIPFSNodeLock() *IPFSNodeLock {
  71. compTable := &LockCompatibilityTable{}
  72. ipfsLock := IPFSNodeLock{
  73. lockCompatibilityTable: compTable,
  74. }
  75. compTable.
  76. Column(IPFSBuzyLock, func() bool { return len(ipfsLock.buzyReqIDs) > 0 }).
  77. Column(IPFSGCLock, func() bool { return len(ipfsLock.gcReqIDs) > 0 })
  78. comp := LockCompatible()
  79. uncp := LockUncompatible()
  80. compTable.MustRow(comp, uncp)
  81. compTable.MustRow(uncp, comp)
  82. return &ipfsLock
  83. }
  84. // CanLock 判断这个锁能否锁定成功
  85. func (l *IPFSNodeLock) CanLock(lock distlock.Lock) error {
  86. return l.lockCompatibilityTable.Test(lock)
  87. }
  88. // 锁定
  89. func (l *IPFSNodeLock) Lock(reqID string, lock distlock.Lock) error {
  90. switch lock.Name {
  91. case IPFSBuzyLock:
  92. l.buzyReqIDs = append(l.buzyReqIDs, reqID)
  93. case IPFSGCLock:
  94. l.gcReqIDs = append(l.gcReqIDs, reqID)
  95. default:
  96. return fmt.Errorf("unknow lock name: %s", lock.Name)
  97. }
  98. return nil
  99. }
  100. // 解锁
  101. func (l *IPFSNodeLock) Unlock(reqID string, lock distlock.Lock) error {
  102. switch lock.Name {
  103. case IPFSBuzyLock:
  104. l.buzyReqIDs = lo2.Remove(l.buzyReqIDs, reqID)
  105. case IPFSGCLock:
  106. l.gcReqIDs = lo2.Remove(l.gcReqIDs, reqID)
  107. default:
  108. return fmt.Errorf("unknow lock name: %s", lock.Name)
  109. }
  110. return nil
  111. }

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