navidrome/utils/singleton/singleton_test.go

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

85 lines
2.0 KiB
Go
Raw Normal View History

2021-06-20 02:56:56 +02:00
package singleton_test
import (
2021-06-20 17:45:59 +02:00
"sync"
"sync/atomic"
2021-06-20 02:56:56 +02:00
"testing"
2021-06-20 17:45:59 +02:00
"github.com/google/uuid"
2021-06-20 02:56:56 +02:00
"github.com/navidrome/navidrome/utils/singleton"
2022-07-26 22:47:16 +02:00
. "github.com/onsi/ginkgo/v2"
2021-06-20 02:56:56 +02:00
. "github.com/onsi/gomega"
)
func TestSingleton(t *testing.T) {
RegisterFailHandler(Fail)
RunSpecs(t, "Singleton Suite")
}
var _ = Describe("GetInstance", func() {
2021-06-20 17:45:59 +02:00
type T struct{ id string }
var numInstances int
constructor := func() *T {
2021-06-20 17:45:59 +02:00
numInstances++
return &T{id: uuid.NewString()}
2021-06-20 02:56:56 +02:00
}
It("calls the constructor to create a new instance", func() {
instance := singleton.GetInstance(constructor)
2021-06-20 17:45:59 +02:00
Expect(numInstances).To(Equal(1))
2021-06-20 02:56:56 +02:00
Expect(instance).To(BeAssignableToTypeOf(&T{}))
})
It("does not call the constructor the next time", func() {
instance := singleton.GetInstance(constructor)
newInstance := singleton.GetInstance(constructor)
2021-06-20 02:56:56 +02:00
Expect(newInstance.id).To(Equal(instance.id))
2021-06-20 17:45:59 +02:00
Expect(numInstances).To(Equal(1))
2021-06-20 02:56:56 +02:00
})
It("makes a distinction between a type and its pointer", func() {
instance := singleton.GetInstance(constructor)
newInstance := singleton.GetInstance(func() T {
numInstances++
return T{id: uuid.NewString()}
})
2021-06-20 02:56:56 +02:00
Expect(instance).To(BeAssignableToTypeOf(&T{}))
Expect(newInstance).To(BeAssignableToTypeOf(T{}))
Expect(newInstance.id).ToNot(Equal(instance.id))
Expect(numInstances).To(Equal(2))
2021-06-20 17:45:59 +02:00
})
It("only calls the constructor once when called concurrently", func() {
const maxCalls = 8000
2021-06-20 17:45:59 +02:00
var numCalls int32
start := sync.WaitGroup{}
start.Add(1)
prepare := sync.WaitGroup{}
prepare.Add(maxCalls)
done := sync.WaitGroup{}
done.Add(maxCalls)
numInstances = 0
2021-06-20 17:45:59 +02:00
for i := 0; i < maxCalls; i++ {
go func() {
start.Wait()
singleton.GetInstance(func() struct{ I int } {
numInstances++
return struct{ I int }{I: 1}
})
2021-06-20 17:45:59 +02:00
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))
2021-06-20 02:56:56 +02:00
})
})