mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00
Merge pull request #14704 from chenyahui/mutex_unlock_state
etcdclient: check mutex state in Unlock method of concurrency.Mutex
This commit is contained in:
commit
0bff3ade4d
@ -18,6 +18,7 @@ import (
|
||||
"context"
|
||||
"errors"
|
||||
"fmt"
|
||||
"strings"
|
||||
"sync"
|
||||
|
||||
pb "go.etcd.io/etcd/api/v3/etcdserverpb"
|
||||
@ -27,6 +28,7 @@ import (
|
||||
// ErrLocked is returned by TryLock when Mutex is already locked by another session.
|
||||
var ErrLocked = errors.New("mutex: Locked by another session")
|
||||
var ErrSessionExpired = errors.New("mutex: session is expired")
|
||||
var ErrLockReleased = errors.New("mutex: lock has already been released")
|
||||
|
||||
// Mutex implements the sync Locker interface with etcd
|
||||
type Mutex struct {
|
||||
@ -128,6 +130,14 @@ func (m *Mutex) tryAcquire(ctx context.Context) (*v3.TxnResponse, error) {
|
||||
}
|
||||
|
||||
func (m *Mutex) Unlock(ctx context.Context) error {
|
||||
if m.myKey == "" || m.myRev <= 0 || m.myKey == "\x00" {
|
||||
return ErrLockReleased
|
||||
}
|
||||
|
||||
if !strings.HasPrefix(m.myKey, m.pfx) {
|
||||
return fmt.Errorf("invalid key %q, it should have prefix %q", m.myKey, m.pfx)
|
||||
}
|
||||
|
||||
client := m.s.Client()
|
||||
if _, err := client.Delete(ctx, m.myKey); err != nil {
|
||||
return err
|
||||
|
@ -16,6 +16,7 @@ package concurrency_test
|
||||
|
||||
import (
|
||||
"context"
|
||||
"errors"
|
||||
"testing"
|
||||
|
||||
"go.etcd.io/etcd/client/v3"
|
||||
@ -70,3 +71,42 @@ func TestMutexLockSessionExpired(t *testing.T) {
|
||||
|
||||
<-m2Locked
|
||||
}
|
||||
|
||||
func TestMutexUnlock(t *testing.T) {
|
||||
cli, err := integration2.NewClient(t, clientv3.Config{Endpoints: exampleEndpoints()})
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer cli.Close()
|
||||
|
||||
s1, err := concurrency.NewSession(cli)
|
||||
if err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
defer s1.Close()
|
||||
|
||||
m1 := concurrency.NewMutex(s1, "/my-lock/")
|
||||
err = m1.Unlock(context.TODO())
|
||||
if err == nil {
|
||||
t.Fatal("expect lock released error")
|
||||
}
|
||||
if !errors.Is(err, concurrency.ErrLockReleased) {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
if err := m1.Lock(context.TODO()); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
if err := m1.Unlock(context.TODO()); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
err = m1.Unlock(context.TODO())
|
||||
if err == nil {
|
||||
t.Fatal("expect lock released error")
|
||||
}
|
||||
if !errors.Is(err, concurrency.ErrLockReleased) {
|
||||
t.Fatal(err)
|
||||
}
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user