2019-11-21 01:33:13 +03:00
|
|
|
package mojangtextures
|
2019-04-21 20:28:58 +03:00
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/elyby/chrly/api/mojang"
|
2021-02-26 02:45:45 +01:00
|
|
|
"github.com/elyby/chrly/utils"
|
2019-04-21 20:28:58 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type inMemoryItem struct {
|
|
|
|
textures *mojang.SignedTexturesResponse
|
|
|
|
timestamp int64
|
|
|
|
}
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
type InMemoryTexturesStorage struct {
|
|
|
|
GCPeriod time.Duration
|
|
|
|
Duration time.Duration
|
|
|
|
|
2020-04-30 00:24:41 +03:00
|
|
|
once sync.Once
|
|
|
|
lock sync.RWMutex
|
|
|
|
data map[string]*inMemoryItem
|
|
|
|
done chan struct{}
|
2019-04-21 20:28:58 +03:00
|
|
|
}
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
func NewInMemoryTexturesStorage() *InMemoryTexturesStorage {
|
|
|
|
storage := &InMemoryTexturesStorage{
|
|
|
|
GCPeriod: 10 * time.Second,
|
|
|
|
Duration: time.Minute + 10*time.Second,
|
|
|
|
data: make(map[string]*inMemoryItem),
|
2019-04-21 20:28:58 +03:00
|
|
|
}
|
2019-06-18 23:34:16 +03:00
|
|
|
|
|
|
|
return storage
|
2019-04-21 20:28:58 +03:00
|
|
|
}
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
func (s *InMemoryTexturesStorage) GetTextures(uuid string) (*mojang.SignedTexturesResponse, error) {
|
2020-01-05 23:45:11 +03:00
|
|
|
s.lock.RLock()
|
|
|
|
defer s.lock.RUnlock()
|
2019-04-21 20:28:58 +03:00
|
|
|
|
|
|
|
item, exists := s.data[uuid]
|
2019-11-21 01:33:13 +03:00
|
|
|
validRange := s.getMinimalNotExpiredTimestamp()
|
2019-05-05 23:06:29 +03:00
|
|
|
if !exists || validRange > item.timestamp {
|
2020-04-28 17:57:51 +03:00
|
|
|
return nil, nil
|
2019-04-21 20:28:58 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return item.textures, nil
|
|
|
|
}
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
func (s *InMemoryTexturesStorage) StoreTextures(uuid string, textures *mojang.SignedTexturesResponse) {
|
2020-04-30 00:24:41 +03:00
|
|
|
s.once.Do(s.start)
|
2019-04-21 20:28:58 +03:00
|
|
|
|
2019-05-05 23:06:29 +03:00
|
|
|
s.lock.Lock()
|
|
|
|
defer s.lock.Unlock()
|
|
|
|
|
|
|
|
s.data[uuid] = &inMemoryItem{
|
2019-04-21 20:28:58 +03:00
|
|
|
textures: textures,
|
2021-02-26 02:45:45 +01:00
|
|
|
timestamp: utils.UnixMillisecond(time.Now()),
|
2019-04-21 20:28:58 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-30 00:24:41 +03:00
|
|
|
func (s *InMemoryTexturesStorage) start() {
|
|
|
|
s.done = make(chan struct{})
|
|
|
|
ticker := time.NewTicker(s.GCPeriod)
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-s.done:
|
|
|
|
return
|
|
|
|
case <-ticker.C:
|
|
|
|
s.gc()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *InMemoryTexturesStorage) Stop() {
|
|
|
|
close(s.done)
|
|
|
|
}
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
func (s *InMemoryTexturesStorage) gc() {
|
2019-04-21 20:28:58 +03:00
|
|
|
s.lock.Lock()
|
|
|
|
defer s.lock.Unlock()
|
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
maxTime := s.getMinimalNotExpiredTimestamp()
|
2019-04-21 20:28:58 +03:00
|
|
|
for uuid, value := range s.data {
|
|
|
|
if maxTime > value.timestamp {
|
|
|
|
delete(s.data, uuid)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-05-05 23:06:29 +03:00
|
|
|
|
2019-11-21 01:33:13 +03:00
|
|
|
func (s *InMemoryTexturesStorage) getMinimalNotExpiredTimestamp() int64 {
|
2021-02-26 02:45:45 +01:00
|
|
|
return utils.UnixMillisecond(time.Now().Add(s.Duration * time.Duration(-1)))
|
2019-05-05 23:06:29 +03:00
|
|
|
}
|