mirror of
				https://github.com/gofiber/storage.git
				synced 2025-10-26 09:50:30 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			156 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			156 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package badger
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/dgraph-io/badger/v3"
 | |
| 	"github.com/gofiber/utils/v2"
 | |
| )
 | |
| 
 | |
| // Storage interface that is implemented by storage providers
 | |
| type Storage struct {
 | |
| 	db         *badger.DB
 | |
| 	gcInterval time.Duration
 | |
| 	done       chan struct{}
 | |
| }
 | |
| 
 | |
| // New creates a new memory storage
 | |
| func New(config ...Config) *Storage {
 | |
| 	// Set default config
 | |
| 	cfg := configDefault(config...)
 | |
| 
 | |
| 	// Set options
 | |
| 	opt := cfg.BadgerOptions
 | |
| 
 | |
| 	// Open database
 | |
| 	db, err := badger.Open(opt)
 | |
| 	if err != nil {
 | |
| 		panic(err)
 | |
| 	}
 | |
| 
 | |
| 	if cfg.Reset {
 | |
| 		if err := db.DropAll(); err != nil {
 | |
| 			panic(err)
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Create storage
 | |
| 	store := &Storage{
 | |
| 		db:         db,
 | |
| 		gcInterval: cfg.GCInterval,
 | |
| 		done:       make(chan struct{}),
 | |
| 	}
 | |
| 
 | |
| 	// Start garbage collector
 | |
| 	go store.gc()
 | |
| 
 | |
| 	return store
 | |
| }
 | |
| 
 | |
| // Get value by key
 | |
| func (s *Storage) Get(key string) ([]byte, error) {
 | |
| 	if len(key) <= 0 {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 	var data []byte
 | |
| 	err := s.db.View(func(txn *badger.Txn) error {
 | |
| 		item, err := txn.Get([]byte(key))
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		// item.Value() is only valid within the transaction.
 | |
| 		// We can either copy it ourselves or use the ValueCopy() method.
 | |
| 		// TODO: Benchmark if it's faster to copy + close tx,
 | |
| 		// or to keep the tx open until unmarshalling is done.
 | |
| 		data, err = item.ValueCopy(nil)
 | |
| 		return err
 | |
| 	})
 | |
| 	// If no value was found return false
 | |
| 	if err == badger.ErrKeyNotFound {
 | |
| 		return nil, nil
 | |
| 	}
 | |
| 	return data, err
 | |
| }
 | |
| 
 | |
| // GetWithContext gets value by key.
 | |
| // Note: This method is not used in the current implementation, but is included to satisfy the Storage interface.
 | |
| func (s *Storage) GetWithContext(ctx context.Context, key string) ([]byte, error) {
 | |
| 	return s.Get(key)
 | |
| }
 | |
| 
 | |
| // Set key with value
 | |
| func (s *Storage) Set(key string, val []byte, exp time.Duration) error {
 | |
| 	// Ain't Nobody Got Time For That
 | |
| 	if len(key) <= 0 || len(val) <= 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	entry := badger.NewEntry(utils.UnsafeBytes(key), val)
 | |
| 	if exp != 0 {
 | |
| 		entry.WithTTL(exp)
 | |
| 	}
 | |
| 	return s.db.Update(func(tx *badger.Txn) error {
 | |
| 		return tx.SetEntry(entry)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // SetWithContext sets key with value.
 | |
| // Note: This method is not used in the current implementation, but is included to satisfy the Storage interface.
 | |
| func (s *Storage) SetWithContext(ctx context.Context, key string, val []byte, exp time.Duration) error {
 | |
| 	return s.Set(key, val, exp)
 | |
| }
 | |
| 
 | |
| // Delete key by key
 | |
| func (s *Storage) Delete(key string) error {
 | |
| 	// Ain't Nobody Got Time For That
 | |
| 	if len(key) <= 0 {
 | |
| 		return nil
 | |
| 	}
 | |
| 	return s.db.Update(func(tx *badger.Txn) error {
 | |
| 		return tx.Delete(utils.UnsafeBytes(key))
 | |
| 	})
 | |
| }
 | |
| 
 | |
| // DeleteWithContext deletes key by key.
 | |
| // Note: This method is not used in the current implementation, but is included to satisfy the Storage interface.
 | |
| func (s *Storage) DeleteWithContext(ctx context.Context, key string) error {
 | |
| 	return s.Delete(key)
 | |
| }
 | |
| 
 | |
| // Reset all keys
 | |
| func (s *Storage) Reset() error {
 | |
| 	return s.db.DropAll()
 | |
| }
 | |
| 
 | |
| // ResetWithContext resets all keys.
 | |
| // Note: This method is not used in the current implementation, but is included to satisfy the Storage interface.
 | |
| func (s *Storage) ResetWithContext(ctx context.Context) error {
 | |
| 	return s.db.DropAll()
 | |
| }
 | |
| 
 | |
| // Close the memory storage
 | |
| func (s *Storage) Close() error {
 | |
| 	s.done <- struct{}{}
 | |
| 	return s.db.Close()
 | |
| }
 | |
| 
 | |
| func (s *Storage) gc() {
 | |
| 	ticker := time.NewTicker(s.gcInterval)
 | |
| 	defer ticker.Stop()
 | |
| 
 | |
| 	for {
 | |
| 		select {
 | |
| 		case <-s.done:
 | |
| 			return
 | |
| 		case <-ticker.C:
 | |
| 			_ = s.db.RunValueLogGC(0.7)
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // Return database client
 | |
| func (s *Storage) Conn() *badger.DB {
 | |
| 	return s.db
 | |
| }
 | 
