2023-06-23 10:10:25 -06:00
|
|
|
package graphvent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"testing"
|
|
|
|
"fmt"
|
2023-06-23 20:56:09 -06:00
|
|
|
"encoding/json"
|
2023-06-24 19:48:59 -06:00
|
|
|
"time"
|
2023-06-23 10:10:25 -06:00
|
|
|
)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
func TestNewBaseLockable(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
_, err = NewBaseLockable(ctx, "Test lockable 2", []Lockable{r1})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestRepeatedChildLockable(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
_, err = NewBaseLockable(ctx, "Test lockable 2", []Lockable{r1, r1})
|
2023-06-23 10:10:25 -06:00
|
|
|
if err == nil {
|
2023-06-23 20:56:09 -06:00
|
|
|
t.Fatal("Added the same lockable as a child twice to the same lockable")
|
2023-06-23 10:10:25 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestLockableSelfLock(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = LockLockable(ctx, r1, r1, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_id := states[0].(LockableState).Owner().ID()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_id != r1.ID() {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of self", owner_id)
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = UnlockLockable(ctx, r1, r1, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner := states[0].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner != nil {
|
|
|
|
return nil, fmt.Errorf("r1 is not unowned after unlock: %s", owner.ID())
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
|
|
|
|
fatalErr(t, err)
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestLockableSelfLockTiered(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r2, err := NewBaseLockable(ctx, "Test lockable 2", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r3, err := NewBaseLockable(ctx, "Test lockable 3", []Lockable{r1, r2})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = LockLockable(ctx, r3, r3, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
_, err = UseStates(ctx, []GraphNode{r1, r2, r3}, func(states []NodeState) (interface{}, error) {
|
|
|
|
owner_1_id := states[0].(LockableState).Owner().ID()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_1_id != r3.ID() {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of r3", owner_1_id)
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_2_id := states[1].(LockableState).Owner().ID()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_2_id != r3.ID() {
|
|
|
|
return nil, fmt.Errorf("r2 is owned by %s instead of r3", owner_2_id)
|
|
|
|
}
|
2023-06-23 20:56:09 -06:00
|
|
|
ser, _ := json.MarshalIndent(states, "", " ")
|
|
|
|
fmt.Printf("\n\n%s\n\n", ser)
|
|
|
|
|
2023-06-23 10:10:25 -06:00
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = UnlockLockable(ctx, r3, r3, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1, r2, r3}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_1 := states[0].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_1 != nil {
|
|
|
|
return nil, fmt.Errorf("r1 is not unowned after unlocking: %s", owner_1.ID())
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_2 := states[1].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_2 != nil {
|
|
|
|
return nil, fmt.Errorf("r2 is not unowned after unlocking: %s", owner_2.ID())
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_3 := states[2].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_3 != nil {
|
|
|
|
return nil, fmt.Errorf("r3 is not unowned after unlocking: %s", owner_3.ID())
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
|
|
|
|
fatalErr(t, err)
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestLockableLockOther(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r2, err := NewBaseLockable(ctx, "Test lockable 2", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UpdateStates(ctx, []GraphNode{r2}, func(states []NodeState) ([]NodeState, interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
node_state := states[0].(LockHolderState)
|
|
|
|
err := LockLockable(ctx, r1, r2, node_state)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
2023-06-23 20:56:09 -06:00
|
|
|
return []NodeState{node_state}, nil, nil
|
2023-06-23 10:10:25 -06:00
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_id := states[0].(LockableState).Owner().ID()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_id != r2.ID() {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of r2", owner_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UpdateStates(ctx, []GraphNode{r2}, func(states []NodeState) ([]NodeState, interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
node_state := states[0].(LockHolderState)
|
|
|
|
err := UnlockLockable(ctx, r1, r2, node_state)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
2023-06-23 20:56:09 -06:00
|
|
|
return []NodeState{node_state}, nil, nil
|
2023-06-23 10:10:25 -06:00
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner := states[0].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner != nil {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of r2", owner.ID())
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestLockableLockSimpleConflict(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r2, err := NewBaseLockable(ctx, "Test lockable 2", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = LockLockable(ctx, r1, r1, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UpdateStates(ctx, []GraphNode{r2}, func(states []NodeState) ([]NodeState, interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
node_state := states[0].(LockHolderState)
|
|
|
|
err := LockLockable(ctx, r1, r2, node_state)
|
2023-06-23 10:10:25 -06:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("r2 took r1's lock from itself")
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
return []NodeState{node_state}, nil, nil
|
2023-06-23 10:10:25 -06:00
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner_id := states[0].(LockableState).Owner().ID()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner_id != r1.ID() {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of r1", owner_id)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = UnlockLockable(ctx, r1, r1, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
_, err = UseStates(ctx, []GraphNode{r1}, func(states []NodeState) (interface{}, error) {
|
2023-06-23 20:56:09 -06:00
|
|
|
owner := states[0].(LockableState).Owner()
|
2023-06-23 10:10:25 -06:00
|
|
|
if owner != nil {
|
|
|
|
return nil, fmt.Errorf("r1 is owned by %s instead of r1", owner.ID())
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
func TestLockableLockTieredConflict(t * testing.T) {
|
2023-06-23 10:10:25 -06:00
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r1, err := NewBaseLockable(ctx, "Test lockable 1", []Lockable{})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r2, err := NewBaseLockable(ctx, "Test lockable 2", []Lockable{r1})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
r3, err := NewBaseLockable(ctx, "Test lockable 3", []Lockable{r1})
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = LockLockable(ctx, r2, r2, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-23 20:56:09 -06:00
|
|
|
err = LockLockable(ctx, r3, r3, nil)
|
2023-06-23 10:10:25 -06:00
|
|
|
if err == nil {
|
|
|
|
t.Fatal("Locked r3 which depends on r1 while r2 which depends on r1 is already locked")
|
|
|
|
}
|
|
|
|
}
|
2023-06-24 19:48:59 -06:00
|
|
|
|
|
|
|
func TestLockableSimpleUpdate(t * testing.T) {
|
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l1, err := NewBaseLockable(ctx, "Test Lockable 1", []Lockable{})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
update_channel := l1.UpdateChannel(0)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
SendUpdate(ctx, l1, NewDirectSignal(l1, "test_update"))
|
|
|
|
}()
|
|
|
|
|
|
|
|
(*GraphTester)(t).WaitForValue(ctx, update_channel, "test_update", l1, 100*time.Millisecond, "Didn't receive test_update sent to l1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLockableDownUpdate(t * testing.T) {
|
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l1, err := NewBaseLockable(ctx, "Test Lockable 1", []Lockable{})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l2, err := NewBaseLockable(ctx, "Test Lockable 2", []Lockable{l1})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
_, err = NewBaseLockable(ctx, "Test Lockable 3", []Lockable{l2})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
update_channel := l1.UpdateChannel(0)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
SendUpdate(ctx, l2, NewDownSignal(l2, "test_update"))
|
|
|
|
}()
|
|
|
|
|
|
|
|
(*GraphTester)(t).WaitForValue(ctx, update_channel, "test_update", l2, 100*time.Millisecond, "Didn't receive test_update on l3 sent on l2")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLockableUpUpdate(t * testing.T) {
|
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l1, err := NewBaseLockable(ctx, "Test Lockable 1", []Lockable{})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l2, err := NewBaseLockable(ctx, "Test Lockable 2", []Lockable{l1})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l3, err := NewBaseLockable(ctx, "Test Lockable 3", []Lockable{l2})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
update_channel := l3.UpdateChannel(0)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
SendUpdate(ctx, l2, NewSignal(l2, "test_update"))
|
|
|
|
}()
|
|
|
|
|
|
|
|
(*GraphTester)(t).WaitForValue(ctx, update_channel, "test_update", l2, 100*time.Millisecond, "Didn't receive test_update on l3 sent on l2")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestOwnerNotUpdatedTwice(t * testing.T) {
|
|
|
|
ctx := testContext(t)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l1, err := NewBaseLockable(ctx, "Test Lockable 1", []Lockable{})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
2023-06-25 22:23:57 -06:00
|
|
|
l2, err := NewBaseLockable(ctx, "Test Lockable 2", []Lockable{l1})
|
2023-06-24 19:48:59 -06:00
|
|
|
fatalErr(t, err)
|
|
|
|
|
|
|
|
update_channel := l2.UpdateChannel(0)
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
SendUpdate(ctx, l1, NewSignal(l1, "test_update"))
|
|
|
|
}()
|
|
|
|
|
|
|
|
(*GraphTester)(t).WaitForValue(ctx, update_channel, "test_update", l1, 100*time.Millisecond, "Dicn't received test_update on l2 from l1")
|
|
|
|
(*GraphTester)(t).CheckForNone(update_channel, "Second update received on dependency")
|
|
|
|
}
|