mirror of
https://gitee.com/johng/gf.git
synced 2024-11-30 03:07:45 +08:00
360 lines
9.0 KiB
Go
360 lines
9.0 KiB
Go
// Copyright GoFrame Author(https://goframe.org). All Rights Reserved.
|
|
//
|
|
// This Source Code Form is subject to the terms of the MIT License.
|
|
// If a copy of the MIT was not distributed with this file,
|
|
// You can obtain one at https://github.com/gogf/gf.
|
|
|
|
package gsession
|
|
|
|
import (
|
|
"context"
|
|
"time"
|
|
|
|
"github.com/gogf/gf/v2/container/gmap"
|
|
"github.com/gogf/gf/v2/container/gvar"
|
|
"github.com/gogf/gf/v2/errors/gcode"
|
|
"github.com/gogf/gf/v2/errors/gerror"
|
|
"github.com/gogf/gf/v2/internal/intlog"
|
|
)
|
|
|
|
// Session struct for storing single session data, which is bound to a single request.
|
|
// The Session struct is the interface with user, but the Storage is the underlying adapter designed interface
|
|
// for functionality implements.
|
|
type Session struct {
|
|
id string // Session id. It retrieves the session if id is custom specified.
|
|
ctx context.Context // Context for current session. Please note that, session lives along with context.
|
|
data *gmap.StrAnyMap // Current Session data, which is retrieved from Storage.
|
|
dirty bool // Used to mark session is modified.
|
|
start bool // Used to mark session is started.
|
|
manager *Manager // Parent session Manager.
|
|
|
|
// idFunc is a callback function used for creating custom session id.
|
|
// This is called if session id is empty ever when session starts.
|
|
idFunc func(ttl time.Duration) (id string)
|
|
}
|
|
|
|
// init does the lazy initialization for session, which retrieves the session if session id is specified,
|
|
// or else it creates a new empty session.
|
|
func (s *Session) init() error {
|
|
if s.start {
|
|
return nil
|
|
}
|
|
var err error
|
|
// Session retrieving.
|
|
if s.id != "" {
|
|
// Retrieve stored session data from storage.
|
|
if s.manager.storage != nil {
|
|
s.data, err = s.manager.storage.GetSession(s.ctx, s.id, s.manager.GetTTL())
|
|
if err != nil && err != ErrorDisabled {
|
|
intlog.Errorf(s.ctx, `session restoring failed for id "%s": %+v`, s.id, err)
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
// Session id creation.
|
|
if s.id == "" {
|
|
if s.idFunc != nil {
|
|
// Use custom session id creating function.
|
|
s.id = s.idFunc(s.manager.ttl)
|
|
} else {
|
|
// Use default session id creating function of storage.
|
|
s.id, err = s.manager.storage.New(s.ctx, s.manager.ttl)
|
|
if err != nil && err != ErrorDisabled {
|
|
intlog.Errorf(s.ctx, "create session id failed: %+v", err)
|
|
return err
|
|
}
|
|
// If session storage does not implements id generating functionality,
|
|
// it then uses default session id creating function.
|
|
if s.id == "" {
|
|
s.id = NewSessionId()
|
|
}
|
|
}
|
|
}
|
|
if s.data == nil {
|
|
s.data = gmap.NewStrAnyMap(true)
|
|
}
|
|
s.start = true
|
|
return nil
|
|
}
|
|
|
|
// Close closes current session and updates its ttl in the session manager.
|
|
// If this session is dirty, it also exports it to storage.
|
|
//
|
|
// NOTE that this function must be called ever after a session request done.
|
|
func (s *Session) Close() error {
|
|
if s.manager.storage == nil {
|
|
return nil
|
|
}
|
|
if s.start && s.id != "" {
|
|
size := s.data.Size()
|
|
if s.dirty {
|
|
err := s.manager.storage.SetSession(s.ctx, s.id, s.data, s.manager.ttl)
|
|
if err != nil && err != ErrorDisabled {
|
|
return err
|
|
}
|
|
} else if size > 0 {
|
|
err := s.manager.storage.UpdateTTL(s.ctx, s.id, s.manager.ttl)
|
|
if err != nil && err != ErrorDisabled {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Set sets key-value pair to this session.
|
|
func (s *Session) Set(key string, value interface{}) (err error) {
|
|
if err = s.init(); err != nil {
|
|
return err
|
|
}
|
|
if err = s.manager.storage.Set(s.ctx, s.id, key, value, s.manager.ttl); err != nil {
|
|
if err == ErrorDisabled {
|
|
s.data.Set(key, value)
|
|
} else {
|
|
return err
|
|
}
|
|
}
|
|
s.dirty = true
|
|
return nil
|
|
}
|
|
|
|
// SetMap batch sets the session using map.
|
|
func (s *Session) SetMap(data map[string]interface{}) (err error) {
|
|
if err = s.init(); err != nil {
|
|
return err
|
|
}
|
|
if err = s.manager.storage.SetMap(s.ctx, s.id, data, s.manager.ttl); err != nil {
|
|
if err == ErrorDisabled {
|
|
s.data.Sets(data)
|
|
} else {
|
|
return err
|
|
}
|
|
}
|
|
s.dirty = true
|
|
return nil
|
|
}
|
|
|
|
// Remove removes key along with its value from this session.
|
|
func (s *Session) Remove(keys ...string) (err error) {
|
|
if s.id == "" {
|
|
return nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return err
|
|
}
|
|
for _, key := range keys {
|
|
if err = s.manager.storage.Remove(s.ctx, s.id, key); err != nil {
|
|
if err == ErrorDisabled {
|
|
s.data.Remove(key)
|
|
} else {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
s.dirty = true
|
|
return nil
|
|
}
|
|
|
|
// RemoveAll deletes all key-value pairs from this session.
|
|
func (s *Session) RemoveAll() (err error) {
|
|
if s.id == "" {
|
|
return nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return err
|
|
}
|
|
if err = s.manager.storage.RemoveAll(s.ctx, s.id); err != nil {
|
|
if err != ErrorDisabled {
|
|
return err
|
|
}
|
|
}
|
|
// Remove data from memory.
|
|
if s.data != nil {
|
|
s.data.Clear()
|
|
}
|
|
s.dirty = true
|
|
return nil
|
|
}
|
|
|
|
// Id returns the session id for this session.
|
|
// It creates and returns a new session id if the session id is not passed in initialization.
|
|
func (s *Session) Id() (id string, err error) {
|
|
if err = s.init(); err != nil {
|
|
return "", err
|
|
}
|
|
return s.id, nil
|
|
}
|
|
|
|
// SetId sets custom session before session starts.
|
|
// It returns error if it is called after session starts.
|
|
func (s *Session) SetId(id string) error {
|
|
if s.start {
|
|
return gerror.NewCode(gcode.CodeInvalidOperation, "session already started")
|
|
}
|
|
s.id = id
|
|
return nil
|
|
}
|
|
|
|
// SetIdFunc sets custom session id creating function before session starts.
|
|
// It returns error if it is called after session starts.
|
|
func (s *Session) SetIdFunc(f func(ttl time.Duration) string) error {
|
|
if s.start {
|
|
return gerror.NewCode(gcode.CodeInvalidOperation, "session already started")
|
|
}
|
|
s.idFunc = f
|
|
return nil
|
|
}
|
|
|
|
// Data returns all data as map.
|
|
// Note that it's using value copy internally for concurrent-safe purpose.
|
|
func (s *Session) Data() (sessionData map[string]interface{}, err error) {
|
|
if s.id == "" {
|
|
return map[string]interface{}{}, nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return nil, err
|
|
}
|
|
sessionData, err = s.manager.storage.Data(s.ctx, s.id)
|
|
if err != nil && err != ErrorDisabled {
|
|
intlog.Errorf(s.ctx, `%+v`, err)
|
|
}
|
|
if sessionData != nil {
|
|
return sessionData, nil
|
|
}
|
|
return s.data.Map(), nil
|
|
}
|
|
|
|
// Size returns the size of the session.
|
|
func (s *Session) Size() (size int, err error) {
|
|
if s.id == "" {
|
|
return 0, nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return 0, err
|
|
}
|
|
size, err = s.manager.storage.GetSize(s.ctx, s.id)
|
|
if err != nil && err != ErrorDisabled {
|
|
intlog.Errorf(s.ctx, `%+v`, err)
|
|
}
|
|
if size > 0 {
|
|
return size, nil
|
|
}
|
|
return s.data.Size(), nil
|
|
}
|
|
|
|
// Contains checks whether key exist in the session.
|
|
func (s *Session) Contains(key string) (ok bool, err error) {
|
|
if s.id == "" {
|
|
return false, nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return false, err
|
|
}
|
|
v, err := s.Get(key)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return !v.IsNil(), nil
|
|
}
|
|
|
|
// IsDirty checks whether there's any data changes in the session.
|
|
func (s *Session) IsDirty() bool {
|
|
return s.dirty
|
|
}
|
|
|
|
// Get retrieves session value with given key.
|
|
// It returns `def` if the key does not exist in the session if `def` is given,
|
|
// or else it returns nil.
|
|
func (s *Session) Get(key string, def ...interface{}) (value *gvar.Var, err error) {
|
|
if s.id == "" {
|
|
return nil, nil
|
|
}
|
|
if err = s.init(); err != nil {
|
|
return nil, err
|
|
}
|
|
v, err := s.manager.storage.Get(s.ctx, s.id, key)
|
|
if err != nil && err != ErrorDisabled {
|
|
intlog.Errorf(s.ctx, `%+v`, err)
|
|
return nil, err
|
|
}
|
|
if v != nil {
|
|
return gvar.New(v), nil
|
|
}
|
|
if v = s.data.Get(key); v != nil {
|
|
return gvar.New(v), nil
|
|
}
|
|
if len(def) > 0 {
|
|
return gvar.New(def[0]), nil
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// MustId performs as function Id, but it panics if any error occurs.
|
|
func (s *Session) MustId() string {
|
|
id, err := s.Id()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return id
|
|
}
|
|
|
|
// MustGet performs as function Get, but it panics if any error occurs.
|
|
func (s *Session) MustGet(key string, def ...interface{}) *gvar.Var {
|
|
v, err := s.Get(key, def...)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return v
|
|
}
|
|
|
|
// MustSet performs as function Set, but it panics if any error occurs.
|
|
func (s *Session) MustSet(key string, value interface{}) {
|
|
err := s.Set(key, value)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// MustSetMap performs as function SetMap, but it panics if any error occurs.
|
|
func (s *Session) MustSetMap(data map[string]interface{}) {
|
|
err := s.SetMap(data)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
// MustContains performs as function Contains, but it panics if any error occurs.
|
|
func (s *Session) MustContains(key string) bool {
|
|
b, err := s.Contains(key)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return b
|
|
}
|
|
|
|
// MustData performs as function Data, but it panics if any error occurs.
|
|
func (s *Session) MustData() map[string]interface{} {
|
|
m, err := s.Data()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return m
|
|
}
|
|
|
|
// MustSize performs as function Size, but it panics if any error occurs.
|
|
func (s *Session) MustSize() int {
|
|
size, err := s.Size()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return size
|
|
}
|
|
|
|
// MustRemove performs as function Remove, but it panics if any error occurs.
|
|
func (s *Session) MustRemove(keys ...string) {
|
|
err := s.Remove(keys...)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|