mirror of
https://github.com/astaxie/beego.git
synced 2024-11-05 16:10:54 +00:00
182 lines
5.1 KiB
Go
182 lines
5.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 (
|
|
"context"
|
|
"crypto/aes"
|
|
"crypto/cipher"
|
|
"encoding/json"
|
|
"net/http"
|
|
"net/url"
|
|
"sync"
|
|
)
|
|
|
|
var cookiepder = &CookieProvider{}
|
|
|
|
// CookieSessionStore Cookie SessionStore
|
|
type CookieSessionStore struct {
|
|
sid string
|
|
values map[interface{}]interface{} // session data
|
|
lock sync.RWMutex
|
|
}
|
|
|
|
// Set value to cookie session.
|
|
// the value are encoded as gob with hash block string.
|
|
func (st *CookieSessionStore) Set(ctx context.Context, key, value interface{}) error {
|
|
st.lock.Lock()
|
|
defer st.lock.Unlock()
|
|
st.values[key] = value
|
|
return nil
|
|
}
|
|
|
|
// Get value from cookie session
|
|
func (st *CookieSessionStore) Get(ctx context.Context, key interface{}) interface{} {
|
|
st.lock.RLock()
|
|
defer st.lock.RUnlock()
|
|
if v, ok := st.values[key]; ok {
|
|
return v
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Delete value in cookie session
|
|
func (st *CookieSessionStore) Delete(ctx context.Context, key interface{}) error {
|
|
st.lock.Lock()
|
|
defer st.lock.Unlock()
|
|
delete(st.values, key)
|
|
return nil
|
|
}
|
|
|
|
// Flush Clean all values in cookie session
|
|
func (st *CookieSessionStore) Flush(context.Context) error {
|
|
st.lock.Lock()
|
|
defer st.lock.Unlock()
|
|
st.values = make(map[interface{}]interface{})
|
|
return nil
|
|
}
|
|
|
|
// SessionID Return id of this cookie session
|
|
func (st *CookieSessionStore) SessionID(context.Context) string {
|
|
return st.sid
|
|
}
|
|
|
|
// SessionRelease Write cookie session to http response cookie
|
|
func (st *CookieSessionStore) SessionRelease(ctx context.Context, w http.ResponseWriter) {
|
|
st.lock.Lock()
|
|
encodedCookie, err := encodeCookie(cookiepder.block, cookiepder.config.SecurityKey, cookiepder.config.SecurityName, st.values)
|
|
st.lock.Unlock()
|
|
if err == nil {
|
|
cookie := &http.Cookie{Name: cookiepder.config.CookieName,
|
|
Value: url.QueryEscape(encodedCookie),
|
|
Path: "/",
|
|
HttpOnly: true,
|
|
Secure: cookiepder.config.Secure,
|
|
MaxAge: cookiepder.config.Maxage}
|
|
http.SetCookie(w, cookie)
|
|
}
|
|
}
|
|
|
|
type cookieConfig struct {
|
|
SecurityKey string `json:"securityKey"`
|
|
BlockKey string `json:"blockKey"`
|
|
SecurityName string `json:"securityName"`
|
|
CookieName string `json:"cookieName"`
|
|
Secure bool `json:"secure"`
|
|
Maxage int `json:"maxage"`
|
|
}
|
|
|
|
// CookieProvider Cookie session provider
|
|
type CookieProvider struct {
|
|
maxlifetime int64
|
|
config *cookieConfig
|
|
block cipher.Block
|
|
}
|
|
|
|
// SessionInit Init cookie session provider with max lifetime and config json.
|
|
// maxlifetime is ignored.
|
|
// json config:
|
|
// securityKey - hash string
|
|
// blockKey - gob encode hash string. it's saved as aes crypto.
|
|
// securityName - recognized name in encoded cookie string
|
|
// cookieName - cookie name
|
|
// maxage - cookie max life time.
|
|
func (pder *CookieProvider) SessionInit(ctx context.Context, maxlifetime int64, config string) error {
|
|
pder.config = &cookieConfig{}
|
|
err := json.Unmarshal([]byte(config), pder.config)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if pder.config.BlockKey == "" {
|
|
pder.config.BlockKey = string(generateRandomKey(16))
|
|
}
|
|
if pder.config.SecurityName == "" {
|
|
pder.config.SecurityName = string(generateRandomKey(20))
|
|
}
|
|
pder.block, err = aes.NewCipher([]byte(pder.config.BlockKey))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
pder.maxlifetime = maxlifetime
|
|
return nil
|
|
}
|
|
|
|
// SessionRead Get SessionStore in cooke.
|
|
// decode cooke string to map and put into SessionStore with sid.
|
|
func (pder *CookieProvider) SessionRead(ctx context.Context, sid string) (Store, error) {
|
|
maps, _ := decodeCookie(pder.block,
|
|
pder.config.SecurityKey,
|
|
pder.config.SecurityName,
|
|
sid, pder.maxlifetime)
|
|
if maps == nil {
|
|
maps = make(map[interface{}]interface{})
|
|
}
|
|
rs := &CookieSessionStore{sid: sid, values: maps}
|
|
return rs, nil
|
|
}
|
|
|
|
// SessionExist Cookie session is always existed
|
|
func (pder *CookieProvider) SessionExist(ctx context.Context, sid string) (bool, error) {
|
|
return true, nil
|
|
}
|
|
|
|
// SessionRegenerate Implement method, no used.
|
|
func (pder *CookieProvider) SessionRegenerate(ctx context.Context, oldsid, sid string) (Store, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
// SessionDestroy Implement method, no used.
|
|
func (pder *CookieProvider) SessionDestroy(ctx context.Context, sid string) error {
|
|
return nil
|
|
}
|
|
|
|
// SessionGC Implement method, no used.
|
|
func (pder *CookieProvider) SessionGC(context.Context) {
|
|
}
|
|
|
|
// SessionAll Implement method, return 0.
|
|
func (pder *CookieProvider) SessionAll(context.Context) int {
|
|
return 0
|
|
}
|
|
|
|
// SessionUpdate Implement method, no used.
|
|
func (pder *CookieProvider) SessionUpdate(ctx context.Context, sid string) error {
|
|
return nil
|
|
}
|
|
|
|
func init() {
|
|
Register("cookie", cookiepder)
|
|
}
|