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)
|
|
|
|
}
|