2017-09-24 13:32:16 +02:00
|
|
|
package badger
|
|
|
|
|
|
|
|
import (
|
2018-04-25 04:29:19 +02:00
|
|
|
"bytes"
|
2017-09-24 13:32:16 +02:00
|
|
|
"os"
|
|
|
|
"runtime"
|
2018-04-25 04:29:19 +02:00
|
|
|
"sync/atomic"
|
2018-04-22 12:52:36 +02:00
|
|
|
"time"
|
2017-09-24 13:32:16 +02:00
|
|
|
|
|
|
|
"github.com/kataras/iris/core/errors"
|
|
|
|
"github.com/kataras/iris/sessions"
|
|
|
|
|
|
|
|
"github.com/dgraph-io/badger"
|
2018-10-02 03:05:39 +02:00
|
|
|
"github.com/kataras/golog"
|
2017-09-24 13:32:16 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// DefaultFileMode used as the default database's "fileMode"
|
|
|
|
// for creating the sessions directory path, opening and write the session file.
|
|
|
|
var (
|
2017-11-02 04:54:33 +01:00
|
|
|
DefaultFileMode = 0755
|
2017-09-24 13:32:16 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Database the badger(key-value file-based) session storage.
|
|
|
|
type Database struct {
|
|
|
|
// Service is the underline badger database connection,
|
|
|
|
// it's initialized at `New` or `NewFromDB`.
|
|
|
|
// Can be used to get stats.
|
2017-10-06 00:19:10 +02:00
|
|
|
Service *badger.DB
|
2018-04-25 04:29:19 +02:00
|
|
|
|
|
|
|
closed uint32 // if 1 is closed.
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
var _ sessions.Database = (*Database)(nil)
|
|
|
|
|
2017-09-24 13:32:16 +02:00
|
|
|
// New creates and returns a new badger(key-value file-based) storage
|
|
|
|
// instance based on the "directoryPath".
|
|
|
|
// DirectoryPath should is the directory which the badger database will store the sessions,
|
|
|
|
// i.e ./sessions
|
|
|
|
//
|
|
|
|
// It will remove any old session files.
|
|
|
|
func New(directoryPath string) (*Database, error) {
|
|
|
|
if directoryPath == "" {
|
2018-04-22 12:52:36 +02:00
|
|
|
return nil, errors.New("directoryPath is missing")
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
lindex := directoryPath[len(directoryPath)-1]
|
|
|
|
if lindex != os.PathSeparator && lindex != '/' {
|
|
|
|
directoryPath += string(os.PathSeparator)
|
|
|
|
}
|
|
|
|
// create directories if necessary
|
|
|
|
if err := os.MkdirAll(directoryPath, os.FileMode(DefaultFileMode)); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
opts := badger.DefaultOptions
|
|
|
|
opts.Dir = directoryPath
|
|
|
|
opts.ValueDir = directoryPath
|
|
|
|
|
2017-10-10 16:07:01 +02:00
|
|
|
service, err := badger.Open(opts)
|
2017-09-24 13:32:16 +02:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
golog.Errorf("unable to initialize the badger-based session database: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
return NewFromDB(service), nil
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewFromDB same as `New` but accepts an already-created custom badger connection instead.
|
2018-04-22 12:52:36 +02:00
|
|
|
func NewFromDB(service *badger.DB) *Database {
|
2017-09-24 13:32:16 +02:00
|
|
|
db := &Database{Service: service}
|
|
|
|
|
|
|
|
runtime.SetFinalizer(db, closeDB)
|
2018-04-22 12:52:36 +02:00
|
|
|
return db
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Acquire receives a session's lifetime from the database,
|
|
|
|
// if the return value is LifeTime{} then the session manager sets the life time based on the expiration duration lives in configuration.
|
|
|
|
func (db *Database) Acquire(sid string, expires time.Duration) sessions.LifeTime {
|
2017-10-06 00:19:10 +02:00
|
|
|
txn := db.Service.NewTransaction(true)
|
|
|
|
defer txn.Commit(nil)
|
|
|
|
|
2018-04-25 04:29:19 +02:00
|
|
|
bsid := makePrefix(sid)
|
2018-04-22 12:52:36 +02:00
|
|
|
item, err := txn.Get(bsid)
|
|
|
|
if err == nil {
|
|
|
|
// found, return the expiration.
|
|
|
|
return sessions.LifeTime{Time: time.Unix(int64(item.ExpiresAt()), 0)}
|
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// not found, create an entry with ttl and return an empty lifetime, session manager will do its job.
|
|
|
|
if err != nil {
|
|
|
|
if err == badger.ErrKeyNotFound {
|
|
|
|
// create it and set the expiration, we don't care about the value there.
|
|
|
|
err = txn.SetWithTTL(bsid, bsid, expires)
|
2017-10-06 00:19:10 +02:00
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
if err != nil {
|
|
|
|
golog.Error(err)
|
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
return sessions.LifeTime{} // session manager will handle the rest.
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-08-14 15:29:04 +02:00
|
|
|
// OnUpdateExpiration not implemented here, yet.
|
|
|
|
// Note that this error will not be logged, callers should catch it manually.
|
|
|
|
func (db *Database) OnUpdateExpiration(sid string, newExpires time.Duration) error {
|
|
|
|
return sessions.ErrNotImplemented
|
|
|
|
}
|
|
|
|
|
2018-04-25 04:29:19 +02:00
|
|
|
var delim = byte('_')
|
|
|
|
|
|
|
|
func makePrefix(sid string) []byte {
|
|
|
|
return append([]byte(sid), delim)
|
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
func makeKey(sid, key string) []byte {
|
2018-04-25 04:29:19 +02:00
|
|
|
return append(makePrefix(sid), []byte(key)...)
|
2018-04-22 12:52:36 +02:00
|
|
|
}
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Set sets a key value of a specific session.
|
|
|
|
// Ignore the "immutable".
|
|
|
|
func (db *Database) Set(sid string, lifetime sessions.LifeTime, key string, value interface{}, immutable bool) {
|
|
|
|
valueBytes, err := sessions.DefaultTranscoder.Marshal(value)
|
2017-10-06 00:19:10 +02:00
|
|
|
if err != nil {
|
2018-04-22 12:52:36 +02:00
|
|
|
golog.Error(err)
|
2017-09-24 13:32:16 +02:00
|
|
|
return
|
|
|
|
}
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
err = db.Service.Update(func(txn *badger.Txn) error {
|
2018-04-25 04:29:19 +02:00
|
|
|
dur := lifetime.DurationUntilExpiration()
|
|
|
|
return txn.SetWithTTL(makeKey(sid, key), valueBytes, dur)
|
2018-04-22 12:52:36 +02:00
|
|
|
})
|
2017-10-06 00:19:10 +02:00
|
|
|
|
|
|
|
if err != nil {
|
2018-04-22 12:52:36 +02:00
|
|
|
golog.Error(err)
|
2017-10-06 00:19:10 +02:00
|
|
|
}
|
2018-04-22 12:52:36 +02:00
|
|
|
}
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Get retrieves a session value based on the key.
|
|
|
|
func (db *Database) Get(sid string, key string) (value interface{}) {
|
|
|
|
err := db.Service.View(func(txn *badger.Txn) error {
|
|
|
|
item, err := txn.Get(makeKey(sid, key))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2018-10-02 03:05:39 +02:00
|
|
|
|
|
|
|
// return item.Value(func(valueBytes []byte) {
|
|
|
|
// if err := sessions.DefaultTranscoder.Unmarshal(valueBytes, &value); err != nil {
|
|
|
|
// golog.Error(err)
|
|
|
|
// }
|
|
|
|
// })
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
valueBytes, err := item.Value()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return sessions.DefaultTranscoder.Unmarshal(valueBytes, &value)
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil && err != badger.ErrKeyNotFound {
|
|
|
|
golog.Error(err)
|
|
|
|
return nil
|
2017-10-06 00:19:10 +02:00
|
|
|
}
|
|
|
|
|
2017-09-24 13:32:16 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Visit loops through all session keys and values.
|
|
|
|
func (db *Database) Visit(sid string, cb func(key string, value interface{})) {
|
2018-04-25 04:29:19 +02:00
|
|
|
prefix := makePrefix(sid)
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
txn := db.Service.NewTransaction(false)
|
|
|
|
defer txn.Discard()
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
iter := txn.NewIterator(badger.DefaultIteratorOptions)
|
|
|
|
defer iter.Close()
|
|
|
|
|
|
|
|
for iter.Rewind(); iter.ValidForPrefix(prefix); iter.Next() {
|
|
|
|
item := iter.Item()
|
2018-10-02 03:05:39 +02:00
|
|
|
var value interface{}
|
|
|
|
|
|
|
|
// err := item.Value(func(valueBytes []byte) {
|
|
|
|
// if err := sessions.DefaultTranscoder.Unmarshal(valueBytes, &value); err != nil {
|
|
|
|
// golog.Error(err)
|
|
|
|
// }
|
|
|
|
// })
|
|
|
|
|
|
|
|
// if err != nil {
|
|
|
|
// golog.Error(err)
|
|
|
|
// continue
|
|
|
|
// }
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
valueBytes, err := item.Value()
|
|
|
|
if err != nil {
|
|
|
|
golog.Error(err)
|
|
|
|
continue
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
2018-04-22 12:52:36 +02:00
|
|
|
|
|
|
|
if err = sessions.DefaultTranscoder.Unmarshal(valueBytes, &value); err != nil {
|
|
|
|
golog.Error(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-04-25 04:29:19 +02:00
|
|
|
cb(string(bytes.TrimPrefix(item.Key(), prefix)), value)
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
2018-04-22 12:52:36 +02:00
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
var iterOptionsNoValues = badger.IteratorOptions{
|
|
|
|
PrefetchValues: false,
|
|
|
|
PrefetchSize: 100,
|
|
|
|
Reverse: false,
|
|
|
|
AllVersions: false,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Len returns the length of the session's entries (keys).
|
|
|
|
func (db *Database) Len(sid string) (n int) {
|
2018-04-25 04:29:19 +02:00
|
|
|
prefix := makePrefix(sid)
|
2018-04-22 12:52:36 +02:00
|
|
|
|
|
|
|
txn := db.Service.NewTransaction(false)
|
|
|
|
iter := txn.NewIterator(iterOptionsNoValues)
|
|
|
|
|
|
|
|
for iter.Rewind(); iter.ValidForPrefix(prefix); iter.Next() {
|
|
|
|
n++
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
iter.Close()
|
|
|
|
txn.Discard()
|
|
|
|
return
|
|
|
|
}
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Delete removes a session key value based on its key.
|
|
|
|
func (db *Database) Delete(sid string, key string) (deleted bool) {
|
|
|
|
txn := db.Service.NewTransaction(true)
|
|
|
|
err := txn.Delete(makeKey(sid, key))
|
2017-09-24 13:32:16 +02:00
|
|
|
if err != nil {
|
2018-04-22 12:52:36 +02:00
|
|
|
golog.Error(err)
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
2018-04-22 12:52:36 +02:00
|
|
|
txn.Commit(nil)
|
|
|
|
return err == nil
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Clear removes all session key values but it keeps the session entry.
|
|
|
|
func (db *Database) Clear(sid string) {
|
2018-04-25 04:29:19 +02:00
|
|
|
prefix := makePrefix(sid)
|
2018-04-22 12:52:36 +02:00
|
|
|
|
2017-10-06 00:19:10 +02:00
|
|
|
txn := db.Service.NewTransaction(true)
|
2018-04-22 12:52:36 +02:00
|
|
|
defer txn.Commit(nil)
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
iter := txn.NewIterator(iterOptionsNoValues)
|
|
|
|
defer iter.Close()
|
|
|
|
|
|
|
|
for iter.Rewind(); iter.ValidForPrefix(prefix); iter.Next() {
|
|
|
|
txn.Delete(iter.Item().Key())
|
2017-10-06 00:19:10 +02:00
|
|
|
}
|
2018-04-22 12:52:36 +02:00
|
|
|
}
|
2017-10-06 00:19:10 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// Release destroys the session, it clears and removes the session entry,
|
|
|
|
// session manager will create a new session ID on the next request after this call.
|
|
|
|
func (db *Database) Release(sid string) {
|
|
|
|
// clear all $sid-$key.
|
|
|
|
db.Clear(sid)
|
|
|
|
// and remove the $sid.
|
|
|
|
txn := db.Service.NewTransaction(true)
|
|
|
|
txn.Delete([]byte(sid))
|
|
|
|
txn.Commit(nil)
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Close shutdowns the badger connection.
|
|
|
|
func (db *Database) Close() error {
|
|
|
|
return closeDB(db)
|
|
|
|
}
|
|
|
|
|
|
|
|
func closeDB(db *Database) error {
|
2018-04-25 04:29:19 +02:00
|
|
|
if atomic.LoadUint32(&db.closed) > 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2017-09-24 13:32:16 +02:00
|
|
|
err := db.Service.Close()
|
|
|
|
if err != nil {
|
|
|
|
golog.Warnf("closing the badger connection: %v", err)
|
2018-04-25 04:29:19 +02:00
|
|
|
} else {
|
|
|
|
atomic.StoreUint32(&db.closed, 1)
|
2017-09-24 13:32:16 +02:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|