You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
86 lines
2.1 KiB
86 lines
2.1 KiB
package util
|
|
|
|
import (
|
|
"fmt"
|
|
"math/rand"
|
|
"sync"
|
|
"testing"
|
|
"time"
|
|
)
|
|
|
|
func TestOrderedLock(t *testing.T) {
|
|
lt := NewLockTable[string]()
|
|
|
|
var wg sync.WaitGroup
|
|
// Simulate transactions requesting locks
|
|
for i := 1; i <= 50; i++ {
|
|
wg.Add(1)
|
|
go func(i int) {
|
|
defer wg.Done()
|
|
key := "resource"
|
|
lockType := SharedLock
|
|
if i%5 == 0 {
|
|
lockType = ExclusiveLock
|
|
}
|
|
|
|
// Simulate attempting to acquire the lock
|
|
lock := lt.AcquireLock("", key, lockType)
|
|
|
|
// Lock acquired, perform some work
|
|
fmt.Printf("ActiveLock %d acquired lock %v\n", lock.ID, lockType)
|
|
|
|
// Simulate some work
|
|
time.Sleep(time.Duration(rand.Int31n(10)*10) * time.Millisecond)
|
|
|
|
// Release the lock
|
|
lt.ReleaseLock(key, lock)
|
|
fmt.Printf("ActiveLock %d released lock %v\n", lock.ID, lockType)
|
|
}(i)
|
|
}
|
|
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestShouldWaitForSharedLock(t *testing.T) {
|
|
lock := &ActiveLock{ID: 2, lockType: SharedLock}
|
|
entry := &LockEntry{
|
|
waiters: []*ActiveLock{{ID: 1}, lock},
|
|
}
|
|
|
|
if !shouldWaitForSharedLock(lock, entry) {
|
|
t.Fatal("shared lock should wait behind earlier waiters")
|
|
}
|
|
|
|
entry.waiters = []*ActiveLock{lock}
|
|
entry.activeExclusiveLockOwnerCount = 1
|
|
if !shouldWaitForSharedLock(lock, entry) {
|
|
t.Fatal("shared lock should wait while an exclusive owner is active")
|
|
}
|
|
|
|
lock.isDeleted = true
|
|
if shouldWaitForSharedLock(lock, entry) {
|
|
t.Fatal("deleted shared lock should stop waiting even if an exclusive owner is active")
|
|
}
|
|
}
|
|
|
|
func TestShouldWaitForExclusiveLock(t *testing.T) {
|
|
lock := &ActiveLock{ID: 2, lockType: ExclusiveLock}
|
|
entry := &LockEntry{
|
|
waiters: []*ActiveLock{{ID: 1}, lock},
|
|
}
|
|
|
|
if !shouldWaitForExclusiveLock(lock, entry) {
|
|
t.Fatal("exclusive lock should wait behind earlier waiters")
|
|
}
|
|
|
|
entry.waiters = []*ActiveLock{lock}
|
|
entry.activeSharedLockOwnerCount = 1
|
|
if !shouldWaitForExclusiveLock(lock, entry) {
|
|
t.Fatal("exclusive lock should wait while shared owners are active")
|
|
}
|
|
|
|
lock.isDeleted = true
|
|
if shouldWaitForExclusiveLock(lock, entry) {
|
|
t.Fatal("deleted exclusive lock should stop waiting even if shared owners are active")
|
|
}
|
|
}
|