mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Make the GetInstance concurrent test more readable
This commit is contained in:
parent
6e2be7f95f
commit
6f7b48202e
2 changed files with 33 additions and 14 deletions
|
@ -14,7 +14,7 @@ var (
|
||||||
)
|
)
|
||||||
|
|
||||||
// GetInstance returns an existing instance of object. If it is not yet created, calls `constructor`, stores the
|
// GetInstance returns an existing instance of object. If it is not yet created, calls `constructor`, stores the
|
||||||
// result for future calls and return it
|
// result for future calls and returns it
|
||||||
func GetInstance[T any](constructor func() T) T {
|
func GetInstance[T any](constructor func() T) T {
|
||||||
var v T
|
var v T
|
||||||
name := reflect.TypeOf(v).String()
|
name := reflect.TypeOf(v).String()
|
||||||
|
|
|
@ -53,32 +53,51 @@ var _ = Describe("GetInstance", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("only calls the constructor once when called concurrently", func() {
|
It("only calls the constructor once when called concurrently", func() {
|
||||||
const maxCalls = 80000
|
// This test creates 80000 goroutines that call GetInstance concurrently. If the constructor is called more than once, the test will fail.
|
||||||
var numCalls int32
|
const numCallsToDo = 80000
|
||||||
|
var numCallsDone atomic.Uint32
|
||||||
|
|
||||||
|
// This WaitGroup is used to make sure all goroutines are ready before the test starts
|
||||||
|
prepare := sync.WaitGroup{}
|
||||||
|
prepare.Add(numCallsToDo)
|
||||||
|
|
||||||
|
// This WaitGroup is used to synchronize the start of all goroutines as simultaneous as possible
|
||||||
start := sync.WaitGroup{}
|
start := sync.WaitGroup{}
|
||||||
start.Add(1)
|
start.Add(1)
|
||||||
prepare := sync.WaitGroup{}
|
|
||||||
prepare.Add(maxCalls)
|
// This WaitGroup is used to wait for all goroutines to be done
|
||||||
done := sync.WaitGroup{}
|
done := sync.WaitGroup{}
|
||||||
done.Add(maxCalls)
|
done.Add(numCallsToDo)
|
||||||
|
|
||||||
numInstancesCreated = 0
|
numInstancesCreated = 0
|
||||||
for i := 0; i < maxCalls; i++ {
|
for i := 0; i < numCallsToDo; i++ {
|
||||||
go func() {
|
go func() {
|
||||||
|
// This is needed to make sure the test does not hang if it fails
|
||||||
|
defer GinkgoRecover()
|
||||||
|
|
||||||
|
// Wait for all goroutines to be ready
|
||||||
start.Wait()
|
start.Wait()
|
||||||
singleton.GetInstance(func() struct{ I int } {
|
instance := singleton.GetInstance(func() struct{ I int } {
|
||||||
numInstancesCreated++
|
numInstancesCreated++
|
||||||
return struct{ I int }{I: 1}
|
return struct{ I int }{I: numInstancesCreated}
|
||||||
})
|
})
|
||||||
atomic.AddInt32(&numCalls, 1)
|
// Increment the number of calls done
|
||||||
|
numCallsDone.Add(1)
|
||||||
|
|
||||||
|
// Flag the main WaitGroup that this goroutine is done
|
||||||
done.Done()
|
done.Done()
|
||||||
|
|
||||||
|
// Make sure the instance we get is always the same one
|
||||||
|
Expect(instance.I).To(Equal(1))
|
||||||
}()
|
}()
|
||||||
|
// Flag that this goroutine is ready to start
|
||||||
prepare.Done()
|
prepare.Done()
|
||||||
}
|
}
|
||||||
prepare.Wait()
|
prepare.Wait() // Wait for all goroutines to be ready
|
||||||
start.Done()
|
start.Done() // Start all goroutines
|
||||||
done.Wait()
|
done.Wait() // Wait for all goroutines to be done
|
||||||
|
|
||||||
Expect(numCalls).To(Equal(int32(maxCalls)))
|
Expect(numCallsDone.Load()).To(Equal(uint32(numCallsToDo)))
|
||||||
Expect(numInstancesCreated).To(Equal(1))
|
Expect(numInstancesCreated).To(Equal(1))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue