Skip to content
Snippets Groups Projects
Select Git revision
  • 5db8727a0e07a3064614dbd23e352735fb89da63
  • without_tipselection default
  • develop protected
  • fix/grafana-local-dashboard
  • wasp
  • fix/dashboard-explorer-freeze
  • master
  • feat/timerqueue
  • test/sync_debug_and_650
  • feat/sync_revamp_inv
  • wip/sync
  • tool/db-recovery
  • portcheck/fix
  • fix/synchronization
  • feat/new-dashboard-analysis
  • feat/refactored-analysis-dashboard
  • feat/new-analysis-dashboard
  • test/demo-prometheus-fpc
  • prometheus_metrics
  • wip/analysis-server
  • merge/fpc-test-value-transfer
  • v0.2.2
  • v0.2.1
  • v0.2.0
  • v0.1.3
  • v0.1.2
  • v0.1.1
  • v0.1.0
28 results

database.go

Blame
  • user avatar
    Wolfgang Welz authored
    5db8727a
    History
    database.go 2.88 KiB
    package database
    
    import (
    	"os"
    	"path/filepath"
    	"sync"
    
    	"github.com/dgraph-io/badger"
    	"github.com/dgraph-io/badger/options"
    )
    
    var databasesByName = make(map[string]*databaseImpl)
    var getLock sync.Mutex
    
    var ErrKeyNotFound = badger.ErrKeyNotFound
    
    type databaseImpl struct {
    	db       *badger.DB
    	name     string
    	openLock sync.Mutex
    }
    
    func Get(name string) (Database, error) {
    	getLock.Lock()
    	defer getLock.Unlock()
    
    	if database, exists := databasesByName[name]; exists {
    		return database, nil
    	}
    
    	database := &databaseImpl{
    		db:   nil,
    		name: name,
    	}
    	if err := database.Open(); err != nil {
    		return nil, err
    	}
    
    	databasesByName[name] = database
    
    	return databasesByName[name], nil
    }
    
    func (this *databaseImpl) Open() error {
    	this.openLock.Lock()
    	defer this.openLock.Unlock()
    
    	if this.db == nil {
    		directory := *DIRECTORY.Value
    
    		if _, err := os.Stat(directory); os.IsNotExist(err) {
    			if err := os.Mkdir(directory, 0700); err != nil {
    				return err
    			}
    		}
    
    		opts := badger.DefaultOptions(filepath.Join(directory, this.name))
    
    		opts.Logger = &logger{}
    		opts.Truncate = true
    		opts.TableLoadingMode = options.MemoryMap
    
    		db, err := badger.Open(opts)
    		if err != nil {
    			return err
    		}
    		this.db = db
    	}
    
    	return nil
    }
    
    func (this *databaseImpl) Set(key []byte, value []byte) error {
    	if err := this.db.Update(func(txn *badger.Txn) error { return txn.Set(key, value) }); err != nil {
    		return err
    	}
    
    	return nil
    }
    
    func (this *databaseImpl) Contains(key []byte) (bool, error) {
    	err := this.db.View(func(txn *badger.Txn) error {
    		_, err := txn.Get(key)
    		if err != nil {
    			return err
    		}
    
    		return nil
    	})
    
    	if err == ErrKeyNotFound {
    		return false, nil
    	} else {
    		return err == nil, err
    	}
    }
    
    func (this *databaseImpl) Get(key []byte) ([]byte, error) {
    	var result []byte = nil
    
    	err := this.db.View(func(txn *badger.Txn) error {
    		item, err := txn.Get(key)
    		if err != nil {
    			return err
    		}
    
    		return item.Value(func(val []byte) error {
    			result = append([]byte{}, val...)
    
    			return nil
    		})
    	})
    
    	return result, err
    }
    
    func (this *databaseImpl) Delete(key []byte) error {
    	err := this.db.Update(func(txn *badger.Txn) error {
    		err := txn.Delete(key)
    		return err
    	})
    	return err
    }
    
    func (this *databaseImpl) ForEach(consumer func([]byte, []byte)) error {
    	err := this.db.View(func(txn *badger.Txn) error {
    		// create an iterator the default options
    		it := txn.NewIterator(badger.DefaultIteratorOptions)
    		defer it.Close()
    
    		// loop through every key-value-pair and call the function
    		for it.Rewind(); it.Valid(); it.Next() {
    			item := it.Item()
    
    			value, err := item.ValueCopy(nil)
    			if err != nil {
    				return err
    			}
    
    			consumer(item.Key(), value)
    		}
    		return nil
    	})
    	return err
    }
    
    func (this *databaseImpl) Close() error {
    	this.openLock.Lock()
    	defer this.openLock.Unlock()
    
    	if this.db != nil {
    		err := this.db.Close()
    
    		this.db = nil
    
    		if err != nil {
    			return err
    		}
    	}
    
    	return nil
    }