2020-11-23 07:38:42 +01:00
|
|
|
package limiter
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gofiber/fiber/v2"
|
|
|
|
"github.com/gofiber/fiber/v2/internal/memory"
|
|
|
|
)
|
|
|
|
|
|
|
|
// go:generate msgp
|
|
|
|
// msgp -file="manager.go" -o="manager_msgp.go" -tests=false -unexported
|
|
|
|
// don't forget to replace the msgp import path to:
|
|
|
|
// "github.com/gofiber/fiber/v2/internal/msgp"
|
|
|
|
type item struct {
|
2021-10-18 09:13:53 +02:00
|
|
|
currHits int
|
|
|
|
prevHits int
|
|
|
|
exp uint64
|
2020-11-23 07:38:42 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
//msgp:ignore manager
|
|
|
|
type manager struct {
|
|
|
|
pool sync.Pool
|
|
|
|
memory *memory.Storage
|
|
|
|
storage fiber.Storage
|
|
|
|
}
|
|
|
|
|
|
|
|
func newManager(storage fiber.Storage) *manager {
|
|
|
|
// Create new storage handler
|
|
|
|
manager := &manager{
|
|
|
|
pool: sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return new(item)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
if storage != nil {
|
|
|
|
// Use provided storage if provided
|
|
|
|
manager.storage = storage
|
|
|
|
} else {
|
|
|
|
// Fallback too memory storage
|
|
|
|
manager.memory = memory.New()
|
|
|
|
}
|
|
|
|
return manager
|
|
|
|
}
|
|
|
|
|
|
|
|
// acquire returns an *entry from the sync.Pool
|
|
|
|
func (m *manager) acquire() *item {
|
|
|
|
return m.pool.Get().(*item)
|
|
|
|
}
|
|
|
|
|
|
|
|
// release and reset *entry to sync.Pool
|
|
|
|
func (m *manager) release(e *item) {
|
2021-10-18 09:13:53 +02:00
|
|
|
e.prevHits = 0
|
|
|
|
e.currHits = 0
|
2020-11-23 07:38:42 +01:00
|
|
|
e.exp = 0
|
|
|
|
m.pool.Put(e)
|
|
|
|
}
|
|
|
|
|
|
|
|
// get data from storage or memory
|
|
|
|
func (m *manager) get(key string) (it *item) {
|
|
|
|
if m.storage != nil {
|
|
|
|
it = m.acquire()
|
|
|
|
if raw, _ := m.storage.Get(key); raw != nil {
|
|
|
|
if _, err := it.UnmarshalMsg(raw); err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if it, _ = m.memory.Get(key).(*item); it == nil {
|
|
|
|
it = m.acquire()
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// get raw data from storage or memory
|
|
|
|
func (m *manager) getRaw(key string) (raw []byte) {
|
|
|
|
if m.storage != nil {
|
|
|
|
raw, _ = m.storage.Get(key)
|
|
|
|
} else {
|
|
|
|
raw, _ = m.memory.Get(key).([]byte)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// set data to storage or memory
|
|
|
|
func (m *manager) set(key string, it *item, exp time.Duration) {
|
|
|
|
if m.storage != nil {
|
|
|
|
if raw, err := it.MarshalMsg(nil); err == nil {
|
|
|
|
_ = m.storage.Set(key, raw, exp)
|
|
|
|
}
|
|
|
|
// we can release data because it's serialized to database
|
|
|
|
m.release(it)
|
|
|
|
} else {
|
|
|
|
m.memory.Set(key, it, exp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// set data to storage or memory
|
|
|
|
func (m *manager) setRaw(key string, raw []byte, exp time.Duration) {
|
|
|
|
if m.storage != nil {
|
|
|
|
_ = m.storage.Set(key, raw, exp)
|
|
|
|
} else {
|
|
|
|
m.memory.Set(key, raw, exp)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// delete data from storage or memory
|
|
|
|
func (m *manager) delete(key string) {
|
|
|
|
if m.storage != nil {
|
|
|
|
_ = m.storage.Delete(key)
|
|
|
|
} else {
|
|
|
|
m.memory.Delete(key)
|
|
|
|
}
|
|
|
|
}
|