mirror of
https://github.com/elyby/chrly.git
synced 2025-01-25 04:53:41 +05:30
dac3ca9001
Introduce universal profile entity Remove fs-based capes serving Rework management API Rework Redis storage schema Reducing amount of the bus emitter usage
68 lines
1.8 KiB
Go
68 lines
1.8 KiB
Go
package mojang
|
|
|
|
import (
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/jellydator/ttlcache/v3"
|
|
)
|
|
|
|
type MojangApiTexturesProvider struct {
|
|
MojangApiTexturesEndpoint func(uuid string, signed bool) (*ProfileResponse, error)
|
|
}
|
|
|
|
func (ctx *MojangApiTexturesProvider) GetTextures(uuid string) (*ProfileResponse, error) {
|
|
return ctx.MojangApiTexturesEndpoint(uuid, true)
|
|
}
|
|
|
|
// Perfectly there should be an object with provider and cache implementation,
|
|
// but I decided not to introduce a layer and just implement cache in place.
|
|
type TexturesProviderWithInMemoryCache struct {
|
|
provider TexturesProvider
|
|
once sync.Once
|
|
cache *ttlcache.Cache[string, *ProfileResponse]
|
|
}
|
|
|
|
func NewTexturesProviderWithInMemoryCache(provider TexturesProvider) *TexturesProviderWithInMemoryCache {
|
|
storage := &TexturesProviderWithInMemoryCache{
|
|
provider: provider,
|
|
cache: ttlcache.New[string, *ProfileResponse](
|
|
ttlcache.WithDisableTouchOnHit[string, *ProfileResponse](),
|
|
// I'm aware of ttlcache.WithLoader(), but it doesn't allow to return an error
|
|
),
|
|
}
|
|
|
|
return storage
|
|
}
|
|
|
|
func (s *TexturesProviderWithInMemoryCache) GetTextures(uuid string) (*ProfileResponse, error) {
|
|
item := s.cache.Get(uuid)
|
|
// Don't check item.IsExpired() since Get function is already did this check
|
|
if item != nil {
|
|
return item.Value(), nil
|
|
}
|
|
|
|
result, err := s.provider.GetTextures(uuid)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
s.cache.Set(uuid, result, time.Minute)
|
|
// Call it only after first set so GC will work more often
|
|
s.startGcOnce()
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func (s *TexturesProviderWithInMemoryCache) StopGC() {
|
|
// If you call the Stop() on a non-started GC, the process will hang trying to close the uninitialized channel
|
|
s.startGcOnce()
|
|
s.cache.Stop()
|
|
}
|
|
|
|
func (s *TexturesProviderWithInMemoryCache) startGcOnce() {
|
|
s.once.Do(func() {
|
|
go s.cache.Start()
|
|
})
|
|
}
|