2017-02-15 19:06:19 +01:00
|
|
|
package sessions
|
|
|
|
|
2017-08-07 05:04:35 +02:00
|
|
|
import (
|
2019-10-24 17:57:05 +02:00
|
|
|
"errors"
|
2020-10-04 15:50:21 +02:00
|
|
|
"reflect"
|
2018-04-22 12:52:36 +02:00
|
|
|
"sync"
|
|
|
|
"time"
|
2017-08-07 05:04:35 +02:00
|
|
|
|
2022-02-18 21:19:33 +01:00
|
|
|
"github.com/kataras/iris/v12/context"
|
2019-10-25 00:27:02 +02:00
|
|
|
"github.com/kataras/iris/v12/core/memstore"
|
2020-08-16 06:07:36 +02:00
|
|
|
|
|
|
|
"github.com/kataras/golog"
|
2017-08-07 05:04:35 +02:00
|
|
|
)
|
|
|
|
|
2018-08-14 15:29:04 +02:00
|
|
|
// ErrNotImplemented is returned when a particular feature is not yet implemented yet.
|
|
|
|
// It can be matched directly, i.e: `isNotImplementedError := sessions.ErrNotImplemented.Equal(err)`.
|
|
|
|
var ErrNotImplemented = errors.New("not implemented yet")
|
|
|
|
|
2017-02-15 19:06:19 +01:00
|
|
|
// Database is the interface which all session databases should implement
|
2017-08-07 05:04:35 +02:00
|
|
|
// By design it doesn't support any type of cookie session like other frameworks.
|
|
|
|
// I want to protect you, believe me.
|
|
|
|
// The scope of the database is to store somewhere the sessions in order to
|
|
|
|
// keep them after restarting the server, nothing more.
|
|
|
|
//
|
2018-04-22 12:52:36 +02:00
|
|
|
// Synchronization are made automatically, you can register one using `UseDatabase`.
|
2017-08-07 05:04:35 +02:00
|
|
|
//
|
2018-04-22 12:52:36 +02:00
|
|
|
// Look the `sessiondb` folder for databases implementations.
|
2017-02-15 19:06:19 +01:00
|
|
|
type Database interface {
|
2020-08-16 06:07:36 +02:00
|
|
|
// SetLogger should inject a logger to this Database.
|
|
|
|
SetLogger(*golog.Logger)
|
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.
|
2023-09-26 20:14:57 +02:00
|
|
|
Acquire(sid string, expires time.Duration) memstore.LifeTime
|
2018-08-14 15:29:04 +02:00
|
|
|
// OnUpdateExpiration should re-set the expiration (ttl) of the session entry inside the database,
|
|
|
|
// it is fired on `ShiftExpiration` and `UpdateExpiration`.
|
|
|
|
// If the database does not support change of ttl then the session entry will be cloned to another one
|
|
|
|
// and the old one will be removed, it depends on the chosen database storage.
|
|
|
|
//
|
|
|
|
// Check of error is required, if error returned then the rest session's keys are not proceed.
|
|
|
|
//
|
2018-08-18 14:01:26 +02:00
|
|
|
// If a database does not support this feature then an `ErrNotImplemented` will be returned instead.
|
2018-08-14 15:29:04 +02:00
|
|
|
OnUpdateExpiration(sid string, newExpires time.Duration) error
|
2018-04-22 12:52:36 +02:00
|
|
|
// Set sets a key value of a specific session.
|
|
|
|
// The "immutable" input argument depends on the store, it may not implement it at all.
|
2020-10-04 15:50:21 +02:00
|
|
|
Set(sid string, key string, value interface{}, ttl time.Duration, immutable bool) error
|
2018-04-22 12:52:36 +02:00
|
|
|
// Get retrieves a session value based on the key.
|
|
|
|
Get(sid string, key string) interface{}
|
2020-10-04 15:50:21 +02:00
|
|
|
// Decode binds the "outPtr" to the value associated to the provided "key".
|
|
|
|
Decode(sid, key string, outPtr interface{}) error
|
2018-04-22 12:52:36 +02:00
|
|
|
// Visit loops through all session keys and values.
|
2020-10-04 15:50:21 +02:00
|
|
|
Visit(sid string, cb func(key string, value interface{})) error
|
2018-04-22 12:52:36 +02:00
|
|
|
// Len returns the length of the session's entries (keys).
|
|
|
|
Len(sid string) int
|
|
|
|
// Delete removes a session key value based on its key.
|
|
|
|
Delete(sid string, key string) (deleted bool)
|
|
|
|
// Clear removes all session key values but it keeps the session entry.
|
2020-10-04 15:50:21 +02:00
|
|
|
Clear(sid string) error
|
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.
|
2020-10-04 15:50:21 +02:00
|
|
|
Release(sid string) error
|
|
|
|
// Close should terminate the database connection. It's called automatically on interrupt signals.
|
|
|
|
Close() error
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2022-02-18 21:19:33 +01:00
|
|
|
// DatabaseRequestHandler is an optional interface that a sessions database
|
|
|
|
// can implement. It contains a single EndRequest method which is fired
|
|
|
|
// on the very end of the request life cycle. It should be used to Flush
|
|
|
|
// any local session's values to the client.
|
|
|
|
type DatabaseRequestHandler interface {
|
|
|
|
EndRequest(ctx *context.Context, session *Session)
|
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
type mem struct {
|
|
|
|
values map[string]*memstore.Store
|
|
|
|
mu sync.RWMutex
|
|
|
|
}
|
2017-08-07 05:04:35 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
var _ Database = (*mem)(nil)
|
2017-08-07 05:04:35 +02:00
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
func newMemDB() Database { return &mem{values: make(map[string]*memstore.Store)} }
|
2017-08-07 05:04:35 +02:00
|
|
|
|
2020-08-16 06:07:36 +02:00
|
|
|
func (s *mem) SetLogger(*golog.Logger) {}
|
|
|
|
|
2023-09-26 20:14:57 +02:00
|
|
|
func (s *mem) Acquire(sid string, expires time.Duration) memstore.LifeTime {
|
2018-04-22 12:52:36 +02:00
|
|
|
s.mu.Lock()
|
|
|
|
s.values[sid] = new(memstore.Store)
|
|
|
|
s.mu.Unlock()
|
2023-09-26 20:14:57 +02:00
|
|
|
return memstore.LifeTime{}
|
2018-04-22 12:52:36 +02:00
|
|
|
}
|
2017-08-07 05:04:35 +02:00
|
|
|
|
2018-08-14 15:29:04 +02:00
|
|
|
// Do nothing, the `LifeTime` of the Session will be managed by the callers automatically on memory-based storage.
|
|
|
|
func (s *mem) OnUpdateExpiration(string, time.Duration) error { return nil }
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
// immutable depends on the store, it may not implement it at all.
|
2020-10-04 15:50:21 +02:00
|
|
|
func (s *mem) Set(sid string, key string, value interface{}, _ time.Duration, immutable bool) error {
|
2018-04-22 12:52:36 +02:00
|
|
|
s.mu.RLock()
|
2022-06-17 21:03:18 +02:00
|
|
|
store, ok := s.values[sid]
|
2018-04-22 12:52:36 +02:00
|
|
|
s.mu.RUnlock()
|
2022-06-17 21:03:18 +02:00
|
|
|
if ok {
|
|
|
|
store.Save(key, value, immutable)
|
|
|
|
}
|
2020-10-04 15:50:21 +02:00
|
|
|
|
|
|
|
return nil
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
func (s *mem) Get(sid string, key string) interface{} {
|
2018-05-09 00:28:26 +02:00
|
|
|
s.mu.RLock()
|
2022-06-17 21:03:18 +02:00
|
|
|
store, ok := s.values[sid]
|
2018-05-09 00:28:26 +02:00
|
|
|
s.mu.RUnlock()
|
2022-06-17 21:03:18 +02:00
|
|
|
if ok {
|
|
|
|
return store.Get(key)
|
|
|
|
}
|
2018-05-09 00:28:26 +02:00
|
|
|
|
2022-06-17 21:03:18 +02:00
|
|
|
return nil
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 15:50:21 +02:00
|
|
|
func (s *mem) Decode(sid string, key string, outPtr interface{}) error {
|
2022-06-17 21:03:18 +02:00
|
|
|
v := s.Get(sid, key)
|
2020-10-04 15:50:21 +02:00
|
|
|
if v != nil {
|
|
|
|
reflect.ValueOf(outPtr).Set(reflect.ValueOf(v))
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *mem) Visit(sid string, cb func(key string, value interface{})) error {
|
2022-06-17 21:03:18 +02:00
|
|
|
s.mu.RLock()
|
|
|
|
store, ok := s.values[sid]
|
|
|
|
s.mu.RUnlock()
|
|
|
|
if ok {
|
|
|
|
store.Visit(cb)
|
|
|
|
}
|
|
|
|
|
2020-10-04 15:50:21 +02:00
|
|
|
return nil
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
func (s *mem) Len(sid string) int {
|
2018-05-09 00:28:26 +02:00
|
|
|
s.mu.RLock()
|
2022-06-17 21:03:18 +02:00
|
|
|
store, ok := s.values[sid]
|
2018-05-09 00:28:26 +02:00
|
|
|
s.mu.RUnlock()
|
2022-06-17 21:03:18 +02:00
|
|
|
if ok {
|
|
|
|
return store.Len()
|
|
|
|
}
|
2018-05-09 00:28:26 +02:00
|
|
|
|
2022-06-17 21:03:18 +02:00
|
|
|
return 0
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
func (s *mem) Delete(sid string, key string) (deleted bool) {
|
|
|
|
s.mu.RLock()
|
2022-06-17 21:03:18 +02:00
|
|
|
store, ok := s.values[sid]
|
2018-04-22 12:52:36 +02:00
|
|
|
s.mu.RUnlock()
|
2022-06-17 21:03:18 +02:00
|
|
|
if ok {
|
|
|
|
deleted = store.Remove(key)
|
|
|
|
}
|
|
|
|
|
2018-04-22 12:52:36 +02:00
|
|
|
return
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 15:50:21 +02:00
|
|
|
func (s *mem) Clear(sid string) error {
|
2022-06-17 21:03:18 +02:00
|
|
|
s.mu.RLock()
|
|
|
|
store, ok := s.values[sid]
|
|
|
|
s.mu.RUnlock()
|
|
|
|
if ok {
|
|
|
|
store.Reset()
|
|
|
|
}
|
2020-10-04 15:50:21 +02:00
|
|
|
|
|
|
|
return nil
|
2017-08-07 05:04:35 +02:00
|
|
|
}
|
|
|
|
|
2020-10-04 15:50:21 +02:00
|
|
|
func (s *mem) Release(sid string) error {
|
2018-04-22 12:52:36 +02:00
|
|
|
s.mu.Lock()
|
|
|
|
delete(s.values, sid)
|
|
|
|
s.mu.Unlock()
|
2020-10-04 15:50:21 +02:00
|
|
|
return nil
|
2017-02-15 19:06:19 +01:00
|
|
|
}
|
2020-10-04 15:50:21 +02:00
|
|
|
|
|
|
|
func (s *mem) Close() error { return nil }
|