|
@ -40,57 +40,42 @@ type LiveLock struct { |
|
|
lc *LockClient |
|
|
lc *LockClient |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
// NewLock creates a lock with a very long duration
|
|
|
|
|
|
func (lc *LockClient) NewLock(key string, owner string) (lock *LiveLock) { |
|
|
|
|
|
return lc.doNewLock(key, lock_manager.MaxDuration, owner) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// StartLock starts a goroutine to lock the key and returns immediately.
|
|
|
|
|
|
func (lc *LockClient) StartLock(key string, owner string) (lock *LiveLock) { |
|
|
|
|
|
|
|
|
// NewShortLivedLock creates a lock with a 5-second duration
|
|
|
|
|
|
func (lc *LockClient) NewShortLivedLock(key string, owner string) (lock *LiveLock) { |
|
|
lock = &LiveLock{ |
|
|
lock = &LiveLock{ |
|
|
key: key, |
|
|
key: key, |
|
|
filer: lc.seedFiler, |
|
|
filer: lc.seedFiler, |
|
|
cancelCh: make(chan struct{}), |
|
|
cancelCh: make(chan struct{}), |
|
|
expireAtNs: time.Now().Add(lock_manager.MaxDuration).UnixNano(), |
|
|
|
|
|
|
|
|
expireAtNs: time.Now().Add(5*time.Second).UnixNano(), |
|
|
grpcDialOption: lc.grpcDialOption, |
|
|
grpcDialOption: lc.grpcDialOption, |
|
|
owner: owner, |
|
|
owner: owner, |
|
|
lc: lc, |
|
|
lc: lc, |
|
|
} |
|
|
} |
|
|
go func() { |
|
|
|
|
|
lock.CreateLock(lock_manager.MaxDuration) |
|
|
|
|
|
lc.keepLock(lock) |
|
|
|
|
|
}() |
|
|
|
|
|
|
|
|
lock.retryUntilLocked(5*time.Second) |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
func (lc *LockClient) doNewLock(key string, lockDuration time.Duration, owner string) (lock *LiveLock) { |
|
|
|
|
|
|
|
|
// StartLock starts a goroutine to lock the key and returns immediately.
|
|
|
|
|
|
func (lc *LockClient) StartLock(key string, owner string) (lock *LiveLock) { |
|
|
lock = &LiveLock{ |
|
|
lock = &LiveLock{ |
|
|
key: key, |
|
|
key: key, |
|
|
filer: lc.seedFiler, |
|
|
filer: lc.seedFiler, |
|
|
cancelCh: make(chan struct{}), |
|
|
cancelCh: make(chan struct{}), |
|
|
expireAtNs: time.Now().Add(lockDuration).UnixNano(), |
|
|
|
|
|
|
|
|
expireAtNs: time.Now().Add(lock_manager.MaxDuration).UnixNano(), |
|
|
grpcDialOption: lc.grpcDialOption, |
|
|
grpcDialOption: lc.grpcDialOption, |
|
|
owner: owner, |
|
|
owner: owner, |
|
|
lc: lc, |
|
|
lc: lc, |
|
|
} |
|
|
} |
|
|
var needRenewal bool |
|
|
|
|
|
if lockDuration > lc.maxLockDuration { |
|
|
|
|
|
lockDuration = lc.maxLockDuration |
|
|
|
|
|
needRenewal = true |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
lock.CreateLock(lockDuration) |
|
|
|
|
|
|
|
|
|
|
|
if needRenewal { |
|
|
|
|
|
go lc.keepLock(lock) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
go func() { |
|
|
|
|
|
lock.retryUntilLocked(lock_manager.MaxDuration) |
|
|
|
|
|
lc.keepLock(lock) |
|
|
|
|
|
}() |
|
|
return |
|
|
return |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
func (lock *LiveLock) CreateLock(lockDuration time.Duration) { |
|
|
|
|
|
|
|
|
func (lock *LiveLock) retryUntilLocked(lockDuration time.Duration) { |
|
|
util.RetryUntil("create lock:"+lock.key, func() error { |
|
|
util.RetryUntil("create lock:"+lock.key, func() error { |
|
|
return lock.DoLock(lockDuration) |
|
|
|
|
|
|
|
|
return lock.AttemptToLock(lockDuration) |
|
|
}, func(err error) (shouldContinue bool) { |
|
|
}, func(err error) (shouldContinue bool) { |
|
|
if err != nil { |
|
|
if err != nil { |
|
|
glog.Warningf("create lock %s: %s", lock.key, err) |
|
|
glog.Warningf("create lock %s: %s", lock.key, err) |
|
@ -99,7 +84,7 @@ func (lock *LiveLock) CreateLock(lockDuration time.Duration) { |
|
|
}) |
|
|
}) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
func (lock *LiveLock) DoLock(lockDuration time.Duration) error { |
|
|
|
|
|
|
|
|
func (lock *LiveLock) AttemptToLock(lockDuration time.Duration) error { |
|
|
errorMessage, err := lock.doLock(lockDuration) |
|
|
errorMessage, err := lock.doLock(lockDuration) |
|
|
if err != nil { |
|
|
if err != nil { |
|
|
time.Sleep(time.Second) |
|
|
time.Sleep(time.Second) |
|
@ -117,11 +102,13 @@ func (lock *LiveLock) IsLocked() bool { |
|
|
return lock!=nil && lock.isLocked |
|
|
return lock!=nil && lock.isLocked |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
func (lock *LiveLock) StopLock() error { |
|
|
|
|
|
close(lock.cancelCh) |
|
|
|
|
|
|
|
|
func (lock *LiveLock) StopShortLivedLock() error { |
|
|
if !lock.isLocked { |
|
|
if !lock.isLocked { |
|
|
return nil |
|
|
return nil |
|
|
} |
|
|
} |
|
|
|
|
|
defer func() { |
|
|
|
|
|
lock.isLocked = false |
|
|
|
|
|
}() |
|
|
return pb.WithFilerClient(false, 0, lock.filer, lock.grpcDialOption, func(client filer_pb.SeaweedFilerClient) error { |
|
|
return pb.WithFilerClient(false, 0, lock.filer, lock.grpcDialOption, func(client filer_pb.SeaweedFilerClient) error { |
|
|
_, err := client.DistributedUnlock(context.Background(), &filer_pb.UnlockRequest{ |
|
|
_, err := client.DistributedUnlock(context.Background(), &filer_pb.UnlockRequest{ |
|
|
Name: lock.key, |
|
|
Name: lock.key, |
|
|