mirror of
https://github.com/kataras/iris.git
synced 2025-12-24 05:17:03 +00:00
Update to 8.2.1 | LevelDB-based session database implemented, Read HISTORY.md
Former-commit-id: 4e341b185aedaaa9a04fdebe43b4364dec59e3c8
This commit is contained in:
190
sessions/sessiondb/leveldb/database.go
Normal file
190
sessions/sessiondb/leveldb/database.go
Normal file
@@ -0,0 +1,190 @@
|
||||
package leveldb
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
|
||||
"runtime"
|
||||
|
||||
"github.com/kataras/golog"
|
||||
"github.com/kataras/iris/core/errors"
|
||||
"github.com/kataras/iris/sessions"
|
||||
|
||||
"github.com/syndtr/goleveldb/leveldb"
|
||||
"github.com/syndtr/goleveldb/leveldb/opt"
|
||||
)
|
||||
|
||||
var (
|
||||
// Options used to open the leveldb database, defaults to leveldb's default values.
|
||||
Options = &opt.Options{}
|
||||
// WriteOptions used to put and delete, defaults to leveldb's default values.
|
||||
WriteOptions = &opt.WriteOptions{}
|
||||
// ReadOptions used to iterate over the database, defaults to leveldb's default values.
|
||||
ReadOptions = &opt.ReadOptions{}
|
||||
)
|
||||
|
||||
// Database the LevelDB(file-based) session storage.
|
||||
type Database struct {
|
||||
// Service is the underline LevelDB database connection,
|
||||
// it's initialized at `New` or `NewFromDB`.
|
||||
// Can be used to get stats.
|
||||
Service *leveldb.DB
|
||||
async bool
|
||||
}
|
||||
|
||||
// New creates and returns a new LevelDB(file-based) storage
|
||||
// instance based on the "directoryPath".
|
||||
// DirectoryPath should is the directory which the leveldb database will store the sessions,
|
||||
// i.e ./sessions/
|
||||
//
|
||||
// It will remove any old session files.
|
||||
func New(directoryPath string) (*Database, error) {
|
||||
|
||||
if directoryPath == "" {
|
||||
return nil, errors.New("dir is missing")
|
||||
}
|
||||
|
||||
// Second parameter is a "github.com/syndtr/goleveldb/leveldb/opt.Options{}"
|
||||
// user can change the `Options` or create the sessiondb via `NewFromDB`
|
||||
// if wants to use a customized leveldb database
|
||||
// or an existing one, we don't require leveldb options at the constructor.
|
||||
//
|
||||
// The leveldb creates the directories, if necessary.
|
||||
service, err := leveldb.OpenFile(directoryPath, Options)
|
||||
|
||||
if err != nil {
|
||||
golog.Errorf("unable to initialize the LevelDB-based session database: %v", err)
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return NewFromDB(service)
|
||||
}
|
||||
|
||||
// NewFromDB same as `New` but accepts an already-created custom boltdb connection instead.
|
||||
func NewFromDB(service *leveldb.DB) (*Database, error) {
|
||||
if service == nil {
|
||||
return nil, errors.New("underline database is missing")
|
||||
}
|
||||
|
||||
db := &Database{Service: service}
|
||||
|
||||
runtime.SetFinalizer(db, closeDB)
|
||||
return db, db.Cleanup()
|
||||
}
|
||||
|
||||
// Cleanup removes any invalid(have expired) session entries,
|
||||
// it's being called automatically on `New` as well.
|
||||
func (db *Database) Cleanup() error {
|
||||
iter := db.Service.NewIterator(nil, ReadOptions)
|
||||
for iter.Next() {
|
||||
// Remember that the contents of the returned slice should not be modified, and
|
||||
// only valid until the next call to Next.
|
||||
k := iter.Key()
|
||||
|
||||
if len(k) > 0 {
|
||||
v := iter.Value()
|
||||
storeDB, err := sessions.DecodeRemoteStore(v)
|
||||
if err != nil {
|
||||
continue
|
||||
}
|
||||
|
||||
if storeDB.Lifetime.HasExpired() {
|
||||
if err := db.Service.Delete(k, WriteOptions); err != nil {
|
||||
golog.Warnf("troubles when cleanup a session remote store from LevelDB: %v", err)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
iter.Release()
|
||||
return iter.Error()
|
||||
}
|
||||
|
||||
// Async if true passed then it will use different
|
||||
// go routines to update the LevelDB(file-based) storage.
|
||||
func (db *Database) Async(useGoRoutines bool) *Database {
|
||||
db.async = useGoRoutines
|
||||
return db
|
||||
}
|
||||
|
||||
// Load loads the sessions from the LevelDB(file-based) session storage.
|
||||
func (db *Database) Load(sid string) (storeDB sessions.RemoteStore) {
|
||||
bsid := []byte(sid)
|
||||
|
||||
iter := db.Service.NewIterator(nil, ReadOptions)
|
||||
for iter.Next() {
|
||||
// Remember that the contents of the returned slice should not be modified, and
|
||||
// only valid until the next call to Next.
|
||||
k := iter.Key()
|
||||
|
||||
if len(k) > 0 {
|
||||
v := iter.Value()
|
||||
if bytes.Equal(k, bsid) { // session id should be the name of the key-value pair
|
||||
store, err := sessions.DecodeRemoteStore(v) // decode the whole value, as a remote store
|
||||
if err != nil {
|
||||
golog.Errorf("error while trying to load from the remote store: %v", err)
|
||||
} else {
|
||||
storeDB = store
|
||||
}
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
iter.Release()
|
||||
if err := iter.Error(); err != nil {
|
||||
golog.Errorf("error while trying to iterate over the database: %v", err)
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
// Sync syncs the database with the session's (memory) store.
|
||||
func (db *Database) Sync(p sessions.SyncPayload) {
|
||||
if db.async {
|
||||
go db.sync(p)
|
||||
} else {
|
||||
db.sync(p)
|
||||
}
|
||||
}
|
||||
|
||||
func (db *Database) sync(p sessions.SyncPayload) {
|
||||
bsid := []byte(p.SessionID)
|
||||
|
||||
if p.Action == sessions.ActionDestroy {
|
||||
if err := db.destroy(bsid); err != nil {
|
||||
golog.Errorf("error while destroying a session(%s) from boltdb: %v",
|
||||
p.SessionID, err)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
s, err := p.Store.Serialize()
|
||||
if err != nil {
|
||||
golog.Errorf("error while serializing the remote store: %v", err)
|
||||
}
|
||||
|
||||
err = db.Service.Put(bsid, s, WriteOptions)
|
||||
|
||||
if err != nil {
|
||||
golog.Errorf("error while writing the session(%s) to the database: %v", p.SessionID, err)
|
||||
}
|
||||
}
|
||||
|
||||
func (db *Database) destroy(bsid []byte) error {
|
||||
return db.Service.Delete(bsid, WriteOptions)
|
||||
}
|
||||
|
||||
// Close shutdowns the LevelDB connection.
|
||||
func (db *Database) Close() error {
|
||||
return closeDB(db)
|
||||
}
|
||||
|
||||
func closeDB(db *Database) error {
|
||||
err := db.Service.Close()
|
||||
if err != nil {
|
||||
golog.Warnf("closing the LevelDB connection: %v", err)
|
||||
}
|
||||
|
||||
return err
|
||||
}
|
||||
Reference in New Issue
Block a user