Fix race condition

This commit is contained in:
Deluan 2024-07-22 14:27:02 -04:00
parent 9b4abd9e5a
commit d046c180bf
2 changed files with 20 additions and 4 deletions

View file

@ -3,6 +3,7 @@ package hasher
import ( import (
"hash/maphash" "hash/maphash"
"strconv" "strconv"
"sync"
"github.com/navidrome/navidrome/utils/random" "github.com/navidrome/navidrome/utils/random"
) )
@ -23,6 +24,7 @@ func HashFunc() func(id, str string) uint64 {
type Hasher struct { type Hasher struct {
seeds map[string]string seeds map[string]string
mutex sync.RWMutex
hashSeed maphash.Seed hashSeed maphash.Seed
} }
@ -35,6 +37,8 @@ func NewHasher() *Hasher {
// SetSeed sets a seed for the given id // SetSeed sets a seed for the given id
func (h *Hasher) SetSeed(id string, seed string) { func (h *Hasher) SetSeed(id string, seed string) {
h.mutex.Lock()
defer h.mutex.Unlock()
h.seeds[id] = seed h.seeds[id] = seed
} }
@ -45,16 +49,17 @@ func (h *Hasher) Reseed(id string) {
func (h *Hasher) reseed(id string) string { func (h *Hasher) reseed(id string) string {
seed := strconv.FormatUint(random.Uint64(), 36) seed := strconv.FormatUint(random.Uint64(), 36)
h.seeds[id] = seed h.SetSeed(id, seed)
return seed return seed
} }
// HashFunc returns a function that hashes a string using the seed for the given id // HashFunc returns a function that hashes a string using the seed for the given id
func (h *Hasher) HashFunc() func(id, str string) uint64 { func (h *Hasher) HashFunc() func(id, str string) uint64 {
return func(id, str string) uint64 { return func(id, str string) uint64 {
var seed string h.mutex.RLock()
var ok bool seed, ok := h.seeds[id]
if seed, ok = h.seeds[id]; !ok { h.mutex.RUnlock()
if !ok {
seed = h.reseed(id) seed = h.reseed(id)
} }

View file

@ -1,6 +1,7 @@
package hasher_test package hasher_test
import ( import (
"strconv"
"testing" "testing"
"github.com/navidrome/navidrome/utils/hasher" "github.com/navidrome/navidrome/utils/hasher"
@ -54,4 +55,14 @@ var _ = Describe("HashFunc", func() {
hasher.SetSeed(id, "original_seed") hasher.SetSeed(id, "original_seed")
Expect(sum).To(Equal(hashFunc(id, input))) Expect(sum).To(Equal(hashFunc(id, input)))
}) })
It("does not cause race conditions", func() {
for i := 0; i < 1000; i++ {
go func() {
hashFunc := hasher.HashFunc()
sum := hashFunc(strconv.Itoa(i), input)
Expect(sum).ToNot(BeZero())
}()
}
})
}) })