mirror of
https://github.com/minio/minio.git
synced 2025-01-11 23:13:23 -05:00
lock: Timeout Unlock RPC call (#12213)
RPC unlock call needs to be timed out otherwise this can block indefinitely. Signed-off-by: Anis Elleuch <anis@min.io>
This commit is contained in:
parent
b81fada834
commit
0b34dfb479
@ -109,7 +109,7 @@ func (l *localLocker) Lock(ctx context.Context, args dsync.LockArgs) (reply bool
|
|||||||
return true, nil
|
return true, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *localLocker) Unlock(args dsync.LockArgs) (reply bool, err error) {
|
func (l *localLocker) Unlock(_ context.Context, args dsync.LockArgs) (reply bool, err error) {
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
|
|
||||||
@ -177,7 +177,7 @@ func (l *localLocker) RLock(ctx context.Context, args dsync.LockArgs) (reply boo
|
|||||||
return reply, nil
|
return reply, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *localLocker) RUnlock(args dsync.LockArgs) (reply bool, err error) {
|
func (l *localLocker) RUnlock(_ context.Context, args dsync.LockArgs) (reply bool, err error) {
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
var lri []lockRequesterInfo
|
var lri []lockRequesterInfo
|
||||||
|
@ -122,8 +122,8 @@ func (client *lockRESTClient) Lock(ctx context.Context, args dsync.LockArgs) (re
|
|||||||
}
|
}
|
||||||
|
|
||||||
// RUnlock calls read unlock REST API.
|
// RUnlock calls read unlock REST API.
|
||||||
func (client *lockRESTClient) RUnlock(args dsync.LockArgs) (reply bool, err error) {
|
func (client *lockRESTClient) RUnlock(ctx context.Context, args dsync.LockArgs) (reply bool, err error) {
|
||||||
return client.restCall(context.Background(), lockRESTMethodRUnlock, args)
|
return client.restCall(ctx, lockRESTMethodRUnlock, args)
|
||||||
}
|
}
|
||||||
|
|
||||||
// RUnlock calls read unlock REST API.
|
// RUnlock calls read unlock REST API.
|
||||||
@ -132,8 +132,8 @@ func (client *lockRESTClient) Refresh(ctx context.Context, args dsync.LockArgs)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Unlock calls write unlock RPC.
|
// Unlock calls write unlock RPC.
|
||||||
func (client *lockRESTClient) Unlock(args dsync.LockArgs) (reply bool, err error) {
|
func (client *lockRESTClient) Unlock(ctx context.Context, args dsync.LockArgs) (reply bool, err error) {
|
||||||
return client.restCall(context.Background(), lockRESTMethodUnlock, args)
|
return client.restCall(ctx, lockRESTMethodUnlock, args)
|
||||||
}
|
}
|
||||||
|
|
||||||
// ForceUnlock calls force unlock handler to forcibly unlock an active lock.
|
// ForceUnlock calls force unlock handler to forcibly unlock an active lock.
|
||||||
|
@ -47,12 +47,12 @@ func TestLockRESTlient(t *testing.T) {
|
|||||||
t.Fatal("Expected for Lock to fail")
|
t.Fatal("Expected for Lock to fail")
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = lkClient.RUnlock(dsync.LockArgs{})
|
_, err = lkClient.RUnlock(context.Background(), dsync.LockArgs{})
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("Expected for RUnlock to fail")
|
t.Fatal("Expected for RUnlock to fail")
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = lkClient.Unlock(dsync.LockArgs{})
|
_, err = lkClient.Unlock(context.Background(), dsync.LockArgs{})
|
||||||
if err == nil {
|
if err == nil {
|
||||||
t.Fatal("Expected for Unlock to fail")
|
t.Fatal("Expected for Unlock to fail")
|
||||||
}
|
}
|
||||||
|
@ -156,7 +156,7 @@ func (l *lockRESTServer) UnlockHandler(w http.ResponseWriter, r *http.Request) {
|
|||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
_, err = l.ll.Unlock(args)
|
_, err = l.ll.Unlock(context.Background(), args)
|
||||||
// Ignore the Unlock() "reply" return value because if err == nil, "reply" is always true
|
// Ignore the Unlock() "reply" return value because if err == nil, "reply" is always true
|
||||||
// Consequently, if err != nil, reply is always false
|
// Consequently, if err != nil, reply is always false
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -203,7 +203,7 @@ func (l *lockRESTServer) RUnlockHandler(w http.ResponseWriter, r *http.Request)
|
|||||||
|
|
||||||
// Ignore the RUnlock() "reply" return value because if err == nil, "reply" is always true.
|
// Ignore the RUnlock() "reply" return value because if err == nil, "reply" is always true.
|
||||||
// Consequently, if err != nil, reply is always false
|
// Consequently, if err != nil, reply is always false
|
||||||
if _, err = l.ll.RUnlock(args); err != nil {
|
if _, err = l.ll.RUnlock(context.Background(), args); err != nil {
|
||||||
l.writeErrorResponse(w, err)
|
l.writeErrorResponse(w, err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -46,7 +46,10 @@ func log(format string, data ...interface{}) {
|
|||||||
const drwMutexAcquireTimeout = 1 * time.Second // 1 second.
|
const drwMutexAcquireTimeout = 1 * time.Second // 1 second.
|
||||||
|
|
||||||
// dRWMutexRefreshTimeout - timeout for the refresh call
|
// dRWMutexRefreshTimeout - timeout for the refresh call
|
||||||
const drwMutexRefreshTimeout = 5 * time.Second
|
const drwMutexRefreshCallTimeout = 5 * time.Second
|
||||||
|
|
||||||
|
// dRWMutexUnlockTimeout - timeout for the unlock call
|
||||||
|
const drwMutexUnlockCallTimeout = 30 * time.Second
|
||||||
|
|
||||||
// dRWMutexRefreshInterval - the interval between two refresh calls
|
// dRWMutexRefreshInterval - the interval between two refresh calls
|
||||||
const drwMutexRefreshInterval = 10 * time.Second
|
const drwMutexRefreshInterval = 10 * time.Second
|
||||||
@ -275,7 +278,7 @@ func refresh(ctx context.Context, ds *Dsync, id, source string, quorum int, lock
|
|||||||
Quorum: quorum,
|
Quorum: quorum,
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx, cancel := context.WithTimeout(ctx, drwMutexRefreshTimeout)
|
ctx, cancel := context.WithTimeout(ctx, drwMutexRefreshCallTimeout)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
refreshed, err := c.Refresh(ctx, args)
|
refreshed, err := c.Refresh(ctx, args)
|
||||||
@ -613,13 +616,16 @@ func sendRelease(ds *Dsync, c NetLocker, owner string, uid string, isReadLock bo
|
|||||||
Resources: names,
|
Resources: names,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), drwMutexUnlockCallTimeout)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
if isReadLock {
|
if isReadLock {
|
||||||
if _, err := c.RUnlock(args); err != nil {
|
if _, err := c.RUnlock(ctx, args); err != nil {
|
||||||
log("dsync: Unable to call RUnlock failed with %s for %#v at %s\n", err, args, c)
|
log("dsync: Unable to call RUnlock failed with %s for %#v at %s\n", err, args, c)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if _, err := c.Unlock(args); err != nil {
|
if _, err := c.Unlock(ctx, args); err != nil {
|
||||||
log("dsync: Unable to call Unlock failed with %s for %#v at %s\n", err, args, c)
|
log("dsync: Unable to call Unlock failed with %s for %#v at %s\n", err, args, c)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
// You should have received a copy of the GNU Affero General Public License
|
// You should have received a copy of the GNU Affero General Public License
|
||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package dsync_test
|
package dsync
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
@ -24,8 +24,6 @@ import (
|
|||||||
"sync/atomic"
|
"sync/atomic"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
. "github.com/minio/minio/pkg/dsync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -15,13 +15,13 @@
|
|||||||
// You should have received a copy of the GNU Affero General Public License
|
// You should have received a copy of the GNU Affero General Public License
|
||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package dsync_test
|
package dsync
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"sync"
|
"sync"
|
||||||
|
"sync/atomic"
|
||||||
. "github.com/minio/minio/pkg/dsync"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
const WriteLock = -1
|
const WriteLock = -1
|
||||||
@ -34,6 +34,9 @@ type lockServer struct {
|
|||||||
|
|
||||||
// Refresh returns lock not found if set to true
|
// Refresh returns lock not found if set to true
|
||||||
lockNotFound bool
|
lockNotFound bool
|
||||||
|
|
||||||
|
// Set to true if you want peers servers to do not respond
|
||||||
|
responseDelay int64
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *lockServer) setRefreshReply(refreshed bool) {
|
func (l *lockServer) setRefreshReply(refreshed bool) {
|
||||||
@ -42,7 +45,15 @@ func (l *lockServer) setRefreshReply(refreshed bool) {
|
|||||||
l.lockNotFound = !refreshed
|
l.lockNotFound = !refreshed
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (l *lockServer) setResponseDelay(responseDelay time.Duration) {
|
||||||
|
atomic.StoreInt64(&l.responseDelay, int64(responseDelay))
|
||||||
|
}
|
||||||
|
|
||||||
func (l *lockServer) Lock(args *LockArgs, reply *bool) error {
|
func (l *lockServer) Lock(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
if _, *reply = l.lockMap[args.Resources[0]]; !*reply {
|
if _, *reply = l.lockMap[args.Resources[0]]; !*reply {
|
||||||
@ -53,6 +64,10 @@ func (l *lockServer) Lock(args *LockArgs, reply *bool) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *lockServer) Unlock(args *LockArgs, reply *bool) error {
|
func (l *lockServer) Unlock(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
var locksHeld int64
|
var locksHeld int64
|
||||||
@ -69,6 +84,10 @@ func (l *lockServer) Unlock(args *LockArgs, reply *bool) error {
|
|||||||
const ReadLock = 1
|
const ReadLock = 1
|
||||||
|
|
||||||
func (l *lockServer) RLock(args *LockArgs, reply *bool) error {
|
func (l *lockServer) RLock(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
var locksHeld int64
|
var locksHeld int64
|
||||||
@ -84,6 +103,10 @@ func (l *lockServer) RLock(args *LockArgs, reply *bool) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *lockServer) RUnlock(args *LockArgs, reply *bool) error {
|
func (l *lockServer) RUnlock(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
var locksHeld int64
|
var locksHeld int64
|
||||||
@ -102,6 +125,10 @@ func (l *lockServer) RUnlock(args *LockArgs, reply *bool) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *lockServer) Refresh(args *LockArgs, reply *bool) error {
|
func (l *lockServer) Refresh(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
*reply = !l.lockNotFound
|
*reply = !l.lockNotFound
|
||||||
@ -109,6 +136,10 @@ func (l *lockServer) Refresh(args *LockArgs, reply *bool) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (l *lockServer) ForceUnlock(args *LockArgs, reply *bool) error {
|
func (l *lockServer) ForceUnlock(args *LockArgs, reply *bool) error {
|
||||||
|
if d := atomic.LoadInt64(&l.responseDelay); d != 0 {
|
||||||
|
time.Sleep(time.Duration(d))
|
||||||
|
}
|
||||||
|
|
||||||
l.mutex.Lock()
|
l.mutex.Lock()
|
||||||
defer l.mutex.Unlock()
|
defer l.mutex.Unlock()
|
||||||
if len(args.UID) != 0 {
|
if len(args.UID) != 0 {
|
||||||
|
@ -15,12 +15,12 @@
|
|||||||
// You should have received a copy of the GNU Affero General Public License
|
// You should have received a copy of the GNU Affero General Public License
|
||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package dsync_test
|
package dsync
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"log"
|
golog "log"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -32,8 +32,6 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/google/uuid"
|
"github.com/google/uuid"
|
||||||
"github.com/minio/minio/pkg/dsync"
|
|
||||||
. "github.com/minio/minio/pkg/dsync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const numberOfNodes = 5
|
const numberOfNodes = 5
|
||||||
@ -56,7 +54,7 @@ func startRPCServers() {
|
|||||||
server.HandleHTTP(rpcPaths[i], fmt.Sprintf("%s-debug", rpcPaths[i]))
|
server.HandleHTTP(rpcPaths[i], fmt.Sprintf("%s-debug", rpcPaths[i]))
|
||||||
l, e := net.Listen("tcp", ":"+strconv.Itoa(i+12345))
|
l, e := net.Listen("tcp", ":"+strconv.Itoa(i+12345))
|
||||||
if e != nil {
|
if e != nil {
|
||||||
log.Fatal("listen error:", e)
|
golog.Fatal("listen error:", e)
|
||||||
}
|
}
|
||||||
go http.Serve(l, nil)
|
go http.Serve(l, nil)
|
||||||
|
|
||||||
@ -244,6 +242,7 @@ func TestFailedRefreshLock(t *testing.T) {
|
|||||||
// Simulate Refresh RPC response to return no locking found
|
// Simulate Refresh RPC response to return no locking found
|
||||||
for i := range lockServers {
|
for i := range lockServers {
|
||||||
lockServers[i].setRefreshReply(false)
|
lockServers[i].setRefreshReply(false)
|
||||||
|
defer lockServers[i].setRefreshReply(true)
|
||||||
}
|
}
|
||||||
|
|
||||||
dm := NewDRWMutex(ds, "aap")
|
dm := NewDRWMutex(ds, "aap")
|
||||||
@ -256,7 +255,7 @@ func TestFailedRefreshLock(t *testing.T) {
|
|||||||
wg.Done()
|
wg.Done()
|
||||||
}
|
}
|
||||||
|
|
||||||
if !dm.GetLock(ctx, cancel, id, source, dsync.Options{Timeout: 5 * time.Minute}) {
|
if !dm.GetLock(ctx, cancel, id, source, Options{Timeout: 5 * time.Minute}) {
|
||||||
t.Fatal("GetLock() should be successful")
|
t.Fatal("GetLock() should be successful")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -268,10 +267,35 @@ func TestFailedRefreshLock(t *testing.T) {
|
|||||||
|
|
||||||
// Should be safe operation in all cases
|
// Should be safe operation in all cases
|
||||||
dm.Unlock()
|
dm.Unlock()
|
||||||
|
}
|
||||||
|
|
||||||
// Revert Refresh RPC response to locking found
|
// Test Unlock should not timeout
|
||||||
|
func TestUnlockShouldNotTimeout(t *testing.T) {
|
||||||
|
dm := NewDRWMutex(ds, "aap")
|
||||||
|
|
||||||
|
if !dm.GetLock(context.Background(), nil, id, source, Options{Timeout: 5 * time.Minute}) {
|
||||||
|
t.Fatal("GetLock() should be successful")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add delay to lock server responses to ensure that lock does not timeout
|
||||||
for i := range lockServers {
|
for i := range lockServers {
|
||||||
lockServers[i].setRefreshReply(false)
|
lockServers[i].setResponseDelay(2 * drwMutexUnlockCallTimeout)
|
||||||
|
defer lockServers[i].setResponseDelay(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
unlockReturned := make(chan struct{}, 1)
|
||||||
|
go func() {
|
||||||
|
dm.Unlock()
|
||||||
|
unlockReturned <- struct{}{}
|
||||||
|
}()
|
||||||
|
|
||||||
|
timer := time.NewTimer(2 * drwMutexUnlockCallTimeout)
|
||||||
|
defer timer.Stop()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-unlockReturned:
|
||||||
|
t.Fatal("Unlock timed out, which should not happen")
|
||||||
|
case <-timer.C:
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,14 +15,12 @@
|
|||||||
// You should have received a copy of the GNU Affero General Public License
|
// You should have received a copy of the GNU Affero General Public License
|
||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
package dsync_test
|
package dsync
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"net/rpc"
|
"net/rpc"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
. "github.com/minio/minio/pkg/dsync"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// ReconnectRPCClient is a wrapper type for rpc.Client which provides reconnect on first failure.
|
// ReconnectRPCClient is a wrapper type for rpc.Client which provides reconnect on first failure.
|
||||||
@ -105,12 +103,12 @@ func (rpcClient *ReconnectRPCClient) Lock(ctx context.Context, args LockArgs) (s
|
|||||||
return status, err
|
return status, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rpcClient *ReconnectRPCClient) RUnlock(args LockArgs) (status bool, err error) {
|
func (rpcClient *ReconnectRPCClient) RUnlock(ctx context.Context, args LockArgs) (status bool, err error) {
|
||||||
err = rpcClient.Call("Dsync.RUnlock", &args, &status)
|
err = rpcClient.Call("Dsync.RUnlock", &args, &status)
|
||||||
return status, err
|
return status, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rpcClient *ReconnectRPCClient) Unlock(args LockArgs) (status bool, err error) {
|
func (rpcClient *ReconnectRPCClient) Unlock(ctx context.Context, args LockArgs) (status bool, err error) {
|
||||||
err = rpcClient.Call("Dsync.Unlock", &args, &status)
|
err = rpcClient.Call("Dsync.Unlock", &args, &status)
|
||||||
return status, err
|
return status, err
|
||||||
}
|
}
|
||||||
|
@ -54,12 +54,16 @@ type NetLocker interface {
|
|||||||
// Do read unlock for given LockArgs. It should return
|
// Do read unlock for given LockArgs. It should return
|
||||||
// * a boolean to indicate success/failure of the operation
|
// * a boolean to indicate success/failure of the operation
|
||||||
// * an error on failure of unlock request operation.
|
// * an error on failure of unlock request operation.
|
||||||
RUnlock(args LockArgs) (bool, error)
|
// Canceling the context will abort the remote call.
|
||||||
|
// In that case, the resource may or may not be unlocked.
|
||||||
|
RUnlock(ctx context.Context, args LockArgs) (bool, error)
|
||||||
|
|
||||||
// Do write unlock for given LockArgs. It should return
|
// Do write unlock for given LockArgs. It should return
|
||||||
// * a boolean to indicate success/failure of the operation
|
// * a boolean to indicate success/failure of the operation
|
||||||
// * an error on failure of unlock request operation.
|
// * an error on failure of unlock request operation.
|
||||||
Unlock(args LockArgs) (bool, error)
|
// Canceling the context will abort the remote call.
|
||||||
|
// In that case, the resource may or may not be unlocked.
|
||||||
|
Unlock(ctx context.Context, args LockArgs) (bool, error)
|
||||||
|
|
||||||
// Refresh the given lock to prevent it from becoming stale
|
// Refresh the given lock to prevent it from becoming stale
|
||||||
Refresh(ctx context.Context, args LockArgs) (bool, error)
|
Refresh(ctx context.Context, args LockArgs) (bool, error)
|
||||||
|
Loading…
Reference in New Issue
Block a user