Add concurrency test for singleton

This commit is contained in:
Deluan 2021-06-20 11:45:59 -04:00
parent 80b2c2f3cf
commit 25db2cb075

View file

@ -1,8 +1,12 @@
package singleton_test
import (
"sync"
"sync/atomic"
"testing"
"github.com/google/uuid"
"github.com/navidrome/navidrome/utils/singleton"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
@ -14,40 +18,58 @@ func TestSingleton(t *testing.T) {
}
var _ = Describe("Get", func() {
type T struct{ val int }
var wasCalled bool
var instance interface{}
type T struct{ id string }
var numInstances int
constructor := func() interface{} {
wasCalled = true
return &T{}
numInstances++
return &T{id: uuid.NewString()}
}
BeforeEach(func() {
instance = singleton.Get(T{}, constructor)
})
It("calls the constructor to create a new instance", func() {
Expect(wasCalled).To(BeTrue())
instance := singleton.Get(T{}, constructor)
Expect(numInstances).To(Equal(1))
Expect(instance).To(BeAssignableToTypeOf(&T{}))
})
It("does not call the constructor the next time", func() {
instance.(*T).val = 10
wasCalled = false
instance := singleton.Get(T{}, constructor)
newInstance := singleton.Get(T{}, constructor)
Expect(newInstance.(*T).val).To(Equal(10))
Expect(wasCalled).To(BeFalse())
Expect(newInstance.(*T).id).To(Equal(instance.(*T).id))
Expect(numInstances).To(Equal(1))
})
It("does not call the constructor even if a pointer is passed as the object", func() {
instance.(*T).val = 20
wasCalled = false
instance := singleton.Get(T{}, constructor)
newInstance := singleton.Get(&T{}, constructor)
Expect(newInstance.(*T).val).To(Equal(20))
Expect(wasCalled).To(BeFalse())
Expect(newInstance.(*T).id).To(Equal(instance.(*T).id))
Expect(numInstances).To(Equal(1))
})
It("only calls the constructor once when called concurrently", func() {
const maxCalls = 2000
var numCalls int32
start := sync.WaitGroup{}
start.Add(1)
prepare := sync.WaitGroup{}
prepare.Add(maxCalls)
done := sync.WaitGroup{}
done.Add(maxCalls)
for i := 0; i < maxCalls; i++ {
go func() {
start.Wait()
singleton.Get(T{}, constructor)
atomic.AddInt32(&numCalls, 1)
done.Done()
}()
prepare.Done()
}
prepare.Wait()
start.Done()
done.Wait()
Expect(numCalls).To(Equal(int32(maxCalls)))
Expect(numInstances).To(Equal(1))
})
})