1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-12 03:07:13 +00:00

302 lines
7.3 KiB
Go
Raw Normal View History

2014-08-18 16:41:43 +08:00
// Copyright 2014 beego Author. All Rights Reserved.
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// 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
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// http://www.apache.org/licenses/LICENSE-2.0
2014-07-03 23:40:21 +08:00
//
2014-08-18 16:41:43 +08:00
// 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 (
"fmt"
"io/ioutil"
2014-01-05 14:48:36 +08:00
"net/http"
"os"
"path"
"path/filepath"
"sync"
"time"
)
var (
filepder = &FileProvider{}
gcmaxlifetime int64
)
2015-09-12 22:53:55 +08:00
// FileSessionStore File session store
type FileSessionStore struct {
sid string
lock sync.RWMutex
values map[interface{}]interface{}
}
// Set value to file session
func (fs *FileSessionStore) Set(key, value interface{}) error {
fs.lock.Lock()
defer fs.lock.Unlock()
fs.values[key] = value
return nil
}
// Get value from file session
func (fs *FileSessionStore) Get(key interface{}) interface{} {
fs.lock.RLock()
defer fs.lock.RUnlock()
if v, ok := fs.values[key]; ok {
return v
}
2015-09-12 22:53:55 +08:00
return nil
}
// Delete value in file session by given key
func (fs *FileSessionStore) Delete(key interface{}) error {
fs.lock.Lock()
defer fs.lock.Unlock()
delete(fs.values, key)
return nil
}
2015-09-12 22:53:55 +08:00
// Flush Clean all values in file session
2013-09-26 18:07:00 +08:00
func (fs *FileSessionStore) Flush() error {
fs.lock.Lock()
defer fs.lock.Unlock()
fs.values = make(map[interface{}]interface{})
return nil
}
2015-09-12 22:53:55 +08:00
// SessionID Get file session store id
func (fs *FileSessionStore) SessionID() string {
return fs.sid
}
2015-09-12 22:53:55 +08:00
// SessionRelease Write file session to local file with Gob string
2014-01-05 14:48:36 +08:00
func (fs *FileSessionStore) SessionRelease(w http.ResponseWriter) {
filepder.lock.Lock()
defer filepder.lock.Unlock()
b, err := EncodeGob(fs.values)
2013-05-14 07:33:00 +08:00
if err != nil {
SLogger.Println(err)
2013-05-14 07:33:00 +08:00
return
}
2014-08-18 14:42:17 +08:00
_, err = os.Stat(path.Join(filepder.savePath, string(fs.sid[0]), string(fs.sid[1]), fs.sid))
var f *os.File
if err == nil {
f, err = os.OpenFile(path.Join(filepder.savePath, string(fs.sid[0]), string(fs.sid[1]), fs.sid), os.O_RDWR, 0777)
2017-04-28 21:38:08 +08:00
if err != nil {
SLogger.Println(err)
return
}
2014-08-18 14:42:17 +08:00
} else if os.IsNotExist(err) {
f, err = os.Create(path.Join(filepder.savePath, string(fs.sid[0]), string(fs.sid[1]), fs.sid))
2017-04-28 21:38:08 +08:00
if err != nil {
SLogger.Println(err)
return
}
2014-08-18 14:42:17 +08:00
} else {
return
}
f.Truncate(0)
f.Seek(0, 0)
f.Write(b)
f.Close()
}
2015-09-12 22:53:55 +08:00
// FileProvider File session provider
type FileProvider struct {
lock sync.RWMutex
maxlifetime int64
savePath string
}
2015-09-12 22:53:55 +08:00
// SessionInit Init file session provider.
// savePath sets the session files path.
func (fp *FileProvider) SessionInit(maxlifetime int64, savePath string) error {
fp.maxlifetime = maxlifetime
fp.savePath = savePath
return nil
}
2015-09-12 22:53:55 +08:00
// SessionRead Read file session by sid.
// if file is not exist, create it.
// the file path is generated from sid string.
2015-09-12 22:53:55 +08:00
func (fp *FileProvider) SessionRead(sid string) (Store, error) {
filepder.lock.Lock()
defer filepder.lock.Unlock()
err := os.MkdirAll(path.Join(fp.savePath, string(sid[0]), string(sid[1])), 0777)
if err != nil {
SLogger.Println(err.Error())
}
_, err = os.Stat(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
var f *os.File
if err == nil {
f, err = os.OpenFile(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid), os.O_RDWR, 0777)
} else if os.IsNotExist(err) {
f, err = os.Create(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
} else {
return nil, err
}
defer f.Close()
os.Chtimes(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid), time.Now(), time.Now())
var kv map[interface{}]interface{}
b, err := ioutil.ReadAll(f)
if err != nil {
return nil, err
}
if len(b) == 0 {
kv = make(map[interface{}]interface{})
} else {
kv, err = DecodeGob(b)
if err != nil {
return nil, err
}
}
2014-08-18 14:42:17 +08:00
ss := &FileSessionStore{sid: sid, values: kv}
return ss, nil
}
2015-09-12 22:53:55 +08:00
// SessionExist Check file session exist.
2017-10-17 17:27:03 +08:00
// it checks the file named from sid exist or not.
2013-11-05 22:23:48 +08:00
func (fp *FileProvider) SessionExist(sid string) bool {
filepder.lock.Lock()
defer filepder.lock.Unlock()
2013-11-05 22:23:48 +08:00
_, err := os.Stat(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
2017-03-17 19:24:45 +02:00
return err == nil
2013-11-05 22:23:48 +08:00
}
2015-09-12 22:53:55 +08:00
// SessionDestroy Remove all files in this save path
func (fp *FileProvider) SessionDestroy(sid string) error {
filepder.lock.Lock()
defer filepder.lock.Unlock()
2014-02-13 18:24:05 +08:00
os.Remove(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
return nil
}
2015-09-12 22:53:55 +08:00
// SessionGC Recycle files in save path
func (fp *FileProvider) SessionGC() {
filepder.lock.Lock()
defer filepder.lock.Unlock()
gcmaxlifetime = fp.maxlifetime
filepath.Walk(fp.savePath, gcpath)
}
2015-09-12 22:53:55 +08:00
// SessionAll Get active file session number.
// it walks save path to count files.
2013-11-02 00:16:10 +08:00
func (fp *FileProvider) SessionAll() int {
a := &activeSession{}
err := filepath.Walk(fp.savePath, func(path string, f os.FileInfo, err error) error {
return a.visit(path, f, err)
})
if err != nil {
SLogger.Printf("filepath.Walk() returned %v\n", err)
2013-11-02 00:16:10 +08:00
return 0
}
return a.total
}
2015-09-12 22:53:55 +08:00
// SessionRegenerate Generate new sid for file session.
// it delete old file and create new file named from new sid.
2015-09-12 22:53:55 +08:00
func (fp *FileProvider) SessionRegenerate(oldsid, sid string) (Store, error) {
filepder.lock.Lock()
defer filepder.lock.Unlock()
oldPath := path.Join(fp.savePath, string(oldsid[0]), string(oldsid[1]))
oldSidFile := path.Join(oldPath, oldsid)
newPath := path.Join(fp.savePath, string(sid[0]), string(sid[1]))
newSidFile := path.Join(newPath, sid)
// new sid file is exist
_, err := os.Stat(newSidFile)
if err == nil {
return nil, fmt.Errorf("newsid %s exist", newSidFile)
2013-09-26 18:07:00 +08:00
}
err = os.MkdirAll(newPath, 0777)
2013-09-26 18:07:00 +08:00
if err != nil {
SLogger.Println(err.Error())
2013-09-26 18:07:00 +08:00
}
// if old sid file exist
// 1.read and parse file content
// 2.write content to new sid file
// 3.remove old sid file, change new sid file atime and ctime
// 4.return FileSessionStore
_, err = os.Stat(oldSidFile)
2013-09-26 18:07:00 +08:00
if err == nil {
b, err := ioutil.ReadFile(oldSidFile)
2013-09-26 18:07:00 +08:00
if err != nil {
return nil, err
}
var kv map[interface{}]interface{}
if len(b) == 0 {
kv = make(map[interface{}]interface{})
} else {
kv, err = DecodeGob(b)
if err != nil {
return nil, err
}
}
ioutil.WriteFile(newSidFile, b, 0777)
os.Remove(oldSidFile)
os.Chtimes(newSidFile, time.Now(), time.Now())
ss := &FileSessionStore{sid: sid, values: kv}
return ss, nil
2013-09-26 18:07:00 +08:00
}
// if old sid file not exist, just create new sid file and return
newf, err := os.Create(newSidFile)
if err != nil {
return nil, err
}
newf.Close()
ss := &FileSessionStore{sid: sid, values: make(map[interface{}]interface{})}
2013-09-26 18:07:00 +08:00
return ss, nil
}
// remove file in save path if expired
func gcpath(path string, info os.FileInfo, err error) error {
if err != nil {
return err
}
if info.IsDir() {
return nil
}
if (info.ModTime().Unix() + gcmaxlifetime) < time.Now().Unix() {
os.Remove(path)
}
return nil
}
2013-11-02 00:16:10 +08:00
type activeSession struct {
total int
}
2015-09-12 22:53:55 +08:00
func (as *activeSession) visit(paths string, f os.FileInfo, err error) error {
2013-11-02 00:16:10 +08:00
if err != nil {
return err
}
if f.IsDir() {
return nil
}
2015-09-12 22:53:55 +08:00
as.total = as.total + 1
2013-11-02 00:16:10 +08:00
return nil
}
func init() {
Register("file", filepder)
}