mirror of
https://github.com/fatedier/frp.git
synced 2025-01-18 23:53:11 +08:00
296 lines
7.1 KiB
Go
296 lines
7.1 KiB
Go
|
// 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 (
|
||
|
"fmt"
|
||
|
"io/ioutil"
|
||
|
"net/http"
|
||
|
"os"
|
||
|
"path"
|
||
|
"path/filepath"
|
||
|
"sync"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
filepder = &FileProvider{}
|
||
|
gcmaxlifetime int64
|
||
|
)
|
||
|
|
||
|
// 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
|
||
|
}
|
||
|
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
|
||
|
}
|
||
|
|
||
|
// Flush Clean all values in file session
|
||
|
func (fs *FileSessionStore) Flush() error {
|
||
|
fs.lock.Lock()
|
||
|
defer fs.lock.Unlock()
|
||
|
fs.values = make(map[interface{}]interface{})
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// SessionID Get file session store id
|
||
|
func (fs *FileSessionStore) SessionID() string {
|
||
|
return fs.sid
|
||
|
}
|
||
|
|
||
|
// SessionRelease Write file session to local file with Gob string
|
||
|
func (fs *FileSessionStore) SessionRelease(w http.ResponseWriter) {
|
||
|
b, err := EncodeGob(fs.values)
|
||
|
if err != nil {
|
||
|
SLogger.Println(err)
|
||
|
return
|
||
|
}
|
||
|
_, 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)
|
||
|
} else if os.IsNotExist(err) {
|
||
|
f, err = os.Create(path.Join(filepder.savePath, string(fs.sid[0]), string(fs.sid[1]), fs.sid))
|
||
|
|
||
|
} else {
|
||
|
return
|
||
|
}
|
||
|
f.Truncate(0)
|
||
|
f.Seek(0, 0)
|
||
|
f.Write(b)
|
||
|
f.Close()
|
||
|
}
|
||
|
|
||
|
// FileProvider File session provider
|
||
|
type FileProvider struct {
|
||
|
lock sync.RWMutex
|
||
|
maxlifetime int64
|
||
|
savePath string
|
||
|
}
|
||
|
|
||
|
// 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
|
||
|
}
|
||
|
|
||
|
// SessionRead Read file session by sid.
|
||
|
// if file is not exist, create it.
|
||
|
// the file path is generated from sid string.
|
||
|
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
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ss := &FileSessionStore{sid: sid, values: kv}
|
||
|
return ss, nil
|
||
|
}
|
||
|
|
||
|
// SessionExist Check file session exist.
|
||
|
// it checkes the file named from sid exist or not.
|
||
|
func (fp *FileProvider) SessionExist(sid string) bool {
|
||
|
filepder.lock.Lock()
|
||
|
defer filepder.lock.Unlock()
|
||
|
|
||
|
_, err := os.Stat(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
|
||
|
if err == nil {
|
||
|
return true
|
||
|
}
|
||
|
return false
|
||
|
}
|
||
|
|
||
|
// SessionDestroy Remove all files in this save path
|
||
|
func (fp *FileProvider) SessionDestroy(sid string) error {
|
||
|
filepder.lock.Lock()
|
||
|
defer filepder.lock.Unlock()
|
||
|
os.Remove(path.Join(fp.savePath, string(sid[0]), string(sid[1]), sid))
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
}
|
||
|
|
||
|
// SessionAll Get active file session number.
|
||
|
// it walks save path to count files.
|
||
|
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)
|
||
|
return 0
|
||
|
}
|
||
|
return a.total
|
||
|
}
|
||
|
|
||
|
// SessionRegenerate Generate new sid for file session.
|
||
|
// it delete old file and create new file named from new sid.
|
||
|
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)
|
||
|
}
|
||
|
|
||
|
err = os.MkdirAll(newPath, 0777)
|
||
|
if err != nil {
|
||
|
SLogger.Println(err.Error())
|
||
|
}
|
||
|
|
||
|
// 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)
|
||
|
if err == nil {
|
||
|
b, err := ioutil.ReadFile(oldSidFile)
|
||
|
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
|
||
|
}
|
||
|
|
||
|
// 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{})}
|
||
|
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
|
||
|
}
|
||
|
|
||
|
type activeSession struct {
|
||
|
total int
|
||
|
}
|
||
|
|
||
|
func (as *activeSession) visit(paths string, f os.FileInfo, err error) error {
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
if f.IsDir() {
|
||
|
return nil
|
||
|
}
|
||
|
as.total = as.total + 1
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func init() {
|
||
|
Register("file", filepder)
|
||
|
}
|