1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-10 11:57:12 +00:00
Beego/server/web/session/sess_mem.go

198 lines
5.5 KiB
Go
Raw Normal View History

2020-07-22 22:50:08 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package session
import (
"container/list"
2020-08-30 15:39:07 +00:00
"context"
2020-07-22 22:50:08 +08:00
"net/http"
"sync"
"time"
)
var mempder = &MemProvider{list: list.New(), sessions: make(map[string]*list.Element)}
// MemSessionStore memory session store.
// it saved sessions in a map in memory.
type MemSessionStore struct {
sid string //session id
timeAccessed time.Time //last access time
value map[interface{}]interface{} //session store
lock sync.RWMutex
}
// Set value to memory session
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) Set(ctx context.Context, key, value interface{}) error {
2020-07-22 22:50:08 +08:00
st.lock.Lock()
defer st.lock.Unlock()
st.value[key] = value
return nil
}
// Get value from memory session by key
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) Get(ctx context.Context, key interface{}) interface{} {
2020-07-22 22:50:08 +08:00
st.lock.RLock()
defer st.lock.RUnlock()
if v, ok := st.value[key]; ok {
return v
}
return nil
}
// Delete in memory session by key
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) Delete(ctx context.Context, key interface{}) error {
2020-07-22 22:50:08 +08:00
st.lock.Lock()
defer st.lock.Unlock()
delete(st.value, key)
return nil
}
// Flush clear all values in memory session
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) Flush(context.Context) error {
2020-07-22 22:50:08 +08:00
st.lock.Lock()
defer st.lock.Unlock()
st.value = make(map[interface{}]interface{})
return nil
}
// SessionID get this id of memory session store
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) SessionID(context.Context) string {
2020-07-22 22:50:08 +08:00
return st.sid
}
// SessionRelease Implement method, no used.
2020-08-30 15:39:07 +00:00
func (st *MemSessionStore) SessionRelease(ctx context.Context, w http.ResponseWriter) {
2020-07-22 22:50:08 +08:00
}
// MemProvider Implement the provider interface
type MemProvider struct {
lock sync.RWMutex // locker
sessions map[string]*list.Element // map in memory
list *list.List // for gc
maxlifetime int64
savePath string
}
// SessionInit init memory session
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionInit(ctx context.Context, maxlifetime int64, savePath string) error {
2020-07-22 22:50:08 +08:00
pder.maxlifetime = maxlifetime
pder.savePath = savePath
return nil
}
// SessionRead get memory session store by sid
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionRead(ctx context.Context, sid string) (Store, error) {
2020-07-22 22:50:08 +08:00
pder.lock.RLock()
if element, ok := pder.sessions[sid]; ok {
2020-09-03 23:36:09 +08:00
go pder.SessionUpdate(nil, sid)
2020-07-22 22:50:08 +08:00
pder.lock.RUnlock()
return element.Value.(*MemSessionStore), nil
}
pder.lock.RUnlock()
pder.lock.Lock()
newsess := &MemSessionStore{sid: sid, timeAccessed: time.Now(), value: make(map[interface{}]interface{})}
element := pder.list.PushFront(newsess)
pder.sessions[sid] = element
pder.lock.Unlock()
return newsess, nil
}
// SessionExist check session store exist in memory session by sid
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionExist(ctx context.Context, sid string) (bool, error) {
2020-07-22 22:50:08 +08:00
pder.lock.RLock()
defer pder.lock.RUnlock()
if _, ok := pder.sessions[sid]; ok {
2020-08-05 18:29:22 +02:00
return true, nil
2020-07-22 22:50:08 +08:00
}
2020-08-05 18:29:22 +02:00
return false, nil
2020-07-22 22:50:08 +08:00
}
// SessionRegenerate generate new sid for session store in memory session
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionRegenerate(ctx context.Context, oldsid, sid string) (Store, error) {
2020-07-22 22:50:08 +08:00
pder.lock.RLock()
if element, ok := pder.sessions[oldsid]; ok {
2020-09-03 23:36:09 +08:00
go pder.SessionUpdate(nil, oldsid)
2020-07-22 22:50:08 +08:00
pder.lock.RUnlock()
pder.lock.Lock()
element.Value.(*MemSessionStore).sid = sid
pder.sessions[sid] = element
delete(pder.sessions, oldsid)
pder.lock.Unlock()
return element.Value.(*MemSessionStore), nil
}
pder.lock.RUnlock()
pder.lock.Lock()
newsess := &MemSessionStore{sid: sid, timeAccessed: time.Now(), value: make(map[interface{}]interface{})}
element := pder.list.PushFront(newsess)
pder.sessions[sid] = element
pder.lock.Unlock()
return newsess, nil
}
// SessionDestroy delete session store in memory session by id
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionDestroy(ctx context.Context, sid string) error {
2020-07-22 22:50:08 +08:00
pder.lock.Lock()
defer pder.lock.Unlock()
if element, ok := pder.sessions[sid]; ok {
delete(pder.sessions, sid)
pder.list.Remove(element)
return nil
}
return nil
}
// SessionGC clean expired session stores in memory session
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionGC(context.Context) {
2020-07-22 22:50:08 +08:00
pder.lock.RLock()
for {
element := pder.list.Back()
if element == nil {
break
}
if (element.Value.(*MemSessionStore).timeAccessed.Unix() + pder.maxlifetime) < time.Now().Unix() {
pder.lock.RUnlock()
pder.lock.Lock()
pder.list.Remove(element)
delete(pder.sessions, element.Value.(*MemSessionStore).sid)
pder.lock.Unlock()
pder.lock.RLock()
} else {
break
}
}
pder.lock.RUnlock()
}
// SessionAll get count number of memory session
2020-08-30 15:39:07 +00:00
func (pder *MemProvider) SessionAll(context.Context) int {
2020-07-22 22:50:08 +08:00
return pder.list.Len()
}
// SessionUpdate expand time of session store by id in memory session
2020-09-03 23:36:09 +08:00
func (pder *MemProvider) SessionUpdate(ctx context.Context, sid string) error {
2020-07-22 22:50:08 +08:00
pder.lock.Lock()
defer pder.lock.Unlock()
if element, ok := pder.sessions[sid]; ok {
element.Value.(*MemSessionStore).timeAccessed = time.Now()
pder.list.MoveToFront(element)
return nil
}
return nil
}
func init() {
Register("memory", mempder)
}