123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132 |
- package cache
- 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
- type item struct {
- body []byte
- ctype []byte
- cencoding []byte
- status int
- exp uint64
- headers map[string][]byte
- // used for finding the item in an indexed heap
- heapidx int
- }
- //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 to 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) //nolint:forcetypeassert // We store nothing else in the pool
- }
- // release and reset *entry to sync.Pool
- func (m *manager) release(e *item) {
- // don't release item if we using memory storage
- if m.storage != nil {
- return
- }
- e.body = nil
- e.ctype = nil
- e.status = 0
- e.exp = 0
- e.headers = nil
- m.pool.Put(e)
- }
- // get data from storage or memory
- func (m *manager) get(key string) *item {
- var it *item
- if m.storage != nil {
- it = m.acquire()
- raw, err := m.storage.Get(key)
- if err != nil {
- return it
- }
- if raw != nil {
- if _, err := it.UnmarshalMsg(raw); err != nil {
- return it
- }
- }
- return it
- }
- if it, _ = m.memory.Get(key).(*item); it == nil { //nolint:errcheck // We store nothing else in the pool
- it = m.acquire()
- return it
- }
- return it
- }
- // get raw data from storage or memory
- func (m *manager) getRaw(key string) []byte {
- var raw []byte
- if m.storage != nil {
- raw, _ = m.storage.Get(key) //nolint:errcheck // TODO: Handle error here
- } else {
- raw, _ = m.memory.Get(key).([]byte) //nolint:errcheck // TODO: Handle error here
- }
- return raw
- }
- // 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) //nolint:errcheck // TODO: Handle error here
- }
- // 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) //nolint:errcheck // TODO: Handle error here
- } else {
- m.memory.Set(key, raw, exp)
- }
- }
- // delete data from storage or memory
- func (m *manager) del(key string) {
- if m.storage != nil {
- _ = m.storage.Delete(key) //nolint:errcheck // TODO: Handle error here
- } else {
- m.memory.Delete(key)
- }
- }
|