1 Star 0 Fork 11

sunstone1997/tango

forked from lunny/tango 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
cookie.go 13.85 KB
一键复制 编辑 原始数据 按行查看 历史
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679
// Copyright 2015 The Tango Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package tango
import (
"bytes"
"crypto/hmac"
"crypto/sha1"
"encoding/base64"
"fmt"
"io/ioutil"
"net/http"
"strconv"
"strings"
"time"
)
func isValidCookieValue(p []byte) bool {
for _, b := range p {
if b <= ' ' ||
b >= 127 ||
b == '"' ||
b == ',' ||
b == ';' ||
b == '\\' {
return false
}
}
return true
}
func isValidCookieName(s string) bool {
for _, r := range s {
if r <= ' ' ||
r >= 127 ||
strings.ContainsRune(" \t\"(),/:;<=>?@[]\\{}", r) {
return false
}
}
return true
}
type Set interface {
String(key string) (string, error)
Int(key string) (int, error)
Int32(key string) (int32, error)
Int64(key string) (int64, error)
Uint(key string) (uint, error)
Uint32(key string) (uint32, error)
Uint64(key string) (uint64, error)
Float32(key string) (float32, error)
Float64(key string) (float64, error)
Bool(key string) (bool, error)
MustString(key string, defs ...string) string
MustInt(key string, defs ...int) int
MustInt32(key string, defs ...int32) int32
MustInt64(key string, defs ...int64) int64
MustUint(key string, defs ...uint) uint
MustUint32(key string, defs ...uint32) uint32
MustUint64(key string, defs ...uint64) uint64
MustFloat32(key string, defs ...float32) float32
MustFloat64(key string, defs ...float64) float64
MustBool(key string, defs ...bool) bool
}
type Cookies interface {
Set
Get(string) *http.Cookie
Set(*http.Cookie)
Expire(string, time.Time)
Del(string)
}
type cookies Context
var _ Cookies = &cookies{}
func NewCookie(name string, value string, age ...int64) *http.Cookie {
if !isValidCookieName(name) || !isValidCookieValue([]byte(value)) {
return nil
}
var utctime time.Time
if len(age) == 0 {
// 2^31 - 1 seconds (roughly 2038)
utctime = time.Unix(2147483647, 0)
} else {
utctime = time.Unix(time.Now().Unix()+age[0], 0)
}
return &http.Cookie{Name: name, Value: value, Expires: utctime}
}
func (c *cookies) Get(key string) *http.Cookie {
ck, err := c.req.Cookie(key)
if err != nil {
return nil
}
return ck
}
func (c *cookies) Set(ck *http.Cookie) {
http.SetCookie(c.ResponseWriter, ck)
}
func (c *cookies) Expire(key string, expire time.Time) {
ck := c.Get(key)
if ck != nil {
ck.Expires = expire
ck.MaxAge = int(expire.Sub(time.Now()).Seconds())
c.Set(ck)
}
}
func (c *cookies) Del(key string) {
c.Expire(key, time.Date(1900, 1, 1, 0, 0, 0, 0, time.Local))
}
func (c *cookies) String(key string) (string, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return "", err
}
return ck.Value, nil
}
func (c *cookies) Int(key string) (int, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
return strconv.Atoi(ck.Value)
}
func (c *cookies) Int32(key string) (int32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
v, err := strconv.ParseInt(ck.Value, 10, 32)
return int32(v), err
}
func (c *cookies) Int64(key string) (int64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
return strconv.ParseInt(ck.Value, 10, 64)
}
func (c *cookies) Uint(key string) (uint, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
v, err := strconv.ParseUint(ck.Value, 10, 64)
return uint(v), err
}
func (c *cookies) Uint32(key string) (uint32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
v, err := strconv.ParseUint(ck.Value, 10, 32)
return uint32(v), err
}
func (c *cookies) Uint64(key string) (uint64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
return strconv.ParseUint(ck.Value, 10, 64)
}
func (c *cookies) Float32(key string) (float32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
v, err := strconv.ParseFloat(ck.Value, 32)
return float32(v), err
}
func (c *cookies) Float64(key string) (float64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
return strconv.ParseFloat(ck.Value, 32)
}
func (c *cookies) Bool(key string) (bool, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return false, err
}
return strconv.ParseBool(ck.Value)
}
func (c *cookies) MustString(key string, defs ...string) string {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return ""
}
return ck.Value
}
func (c *cookies) MustInt(key string, defs ...int) int {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.Atoi(ck.Value)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *cookies) MustInt32(key string, defs ...int32) int32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseInt(ck.Value, 10, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return int32(v)
}
func (c *cookies) MustInt64(key string, defs ...int64) int64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseInt(ck.Value, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *cookies) MustUint(key string, defs ...uint) uint {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseUint(ck.Value, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return uint(v)
}
func (c *cookies) MustUint32(key string, defs ...uint32) uint32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseUint(ck.Value, 10, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return uint32(v)
}
func (c *cookies) MustUint64(key string, defs ...uint64) uint64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseUint(ck.Value, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *cookies) MustFloat32(key string, defs ...float32) float32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseFloat(ck.Value, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return float32(v)
}
func (c *cookies) MustFloat64(key string, defs ...float64) float64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
v, err := strconv.ParseFloat(ck.Value, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *cookies) MustBool(key string, defs ...bool) bool {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return false
}
v, err := strconv.ParseBool(ck.Value)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func getCookieSig(key string, val []byte, timestamp string) string {
hm := hmac.New(sha1.New, []byte(key))
hm.Write(val)
hm.Write([]byte(timestamp))
hex := fmt.Sprintf("%02x", hm.Sum(nil))
return hex
}
// secure cookies
type secureCookies struct {
*cookies
secret string
}
var _ Cookies = &secureCookies{}
func parseSecureCookie(secret string, value string) string {
parts := strings.SplitN(value, "|", 3)
val, timestamp, sig := parts[0], parts[1], parts[2]
if getCookieSig(secret, []byte(val), timestamp) != sig {
return ""
}
ts, _ := strconv.ParseInt(timestamp, 0, 64)
if time.Now().Unix()-31*86400 > ts {
return ""
}
buf := bytes.NewBufferString(val)
encoder := base64.NewDecoder(base64.StdEncoding, buf)
res, _ := ioutil.ReadAll(encoder)
return string(res)
}
func (c *secureCookies) Get(key string) *http.Cookie {
ck := c.cookies.Get(key)
if ck == nil {
return nil
}
v := parseSecureCookie(c.secret, ck.Value)
if v == "" {
return nil
}
ck.Value = v
return ck
}
func (c *secureCookies) String(key string) (string, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return "", err
}
v := parseSecureCookie(c.secret, ck.Value)
return v, nil
}
func (c *secureCookies) Int(key string) (int, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
return strconv.Atoi(s)
}
func (c *secureCookies) Int32(key string) (int32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseInt(s, 10, 32)
return int32(v), err
}
func (c *secureCookies) Int64(key string) (int64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
return strconv.ParseInt(s, 10, 64)
}
func (c *secureCookies) Uint(key string) (uint, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseUint(s, 10, 64)
return uint(v), err
}
func (c *secureCookies) Uint32(key string) (uint32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseUint(s, 10, 32)
return uint32(v), err
}
func (c *secureCookies) Uint64(key string) (uint64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
return strconv.ParseUint(s, 10, 64)
}
func (c *secureCookies) Float32(key string) (float32, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseFloat(s, 32)
return float32(v), err
}
func (c *secureCookies) Float64(key string) (float64, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return 0, err
}
s := parseSecureCookie(c.secret, ck.Value)
return strconv.ParseFloat(s, 32)
}
func (c *secureCookies) Bool(key string) (bool, error) {
ck, err := c.req.Cookie(key)
if err != nil {
return false, err
}
s := parseSecureCookie(c.secret, ck.Value)
return strconv.ParseBool(s)
}
func (c *secureCookies) MustString(key string, defs ...string) string {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return ""
}
s := parseSecureCookie(c.secret, ck.Value)
return s
}
func (c *secureCookies) MustInt(key string, defs ...int) int {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.Atoi(s)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *secureCookies) MustInt32(key string, defs ...int32) int32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseInt(s, 10, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return int32(v)
}
func (c *secureCookies) MustInt64(key string, defs ...int64) int64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseInt(s, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *secureCookies) MustUint(key string, defs ...uint) uint {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseUint(s, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return uint(v)
}
func (c *secureCookies) MustUint32(key string, defs ...uint32) uint32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseUint(s, 10, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return uint32(v)
}
func (c *secureCookies) MustUint64(key string, defs ...uint64) uint64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseUint(s, 10, 64)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *secureCookies) MustFloat32(key string, defs ...float32) float32 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseFloat(s, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return float32(v)
}
func (c *secureCookies) MustFloat64(key string, defs ...float64) float64 {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return 0
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseFloat(s, 32)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func (c *secureCookies) MustBool(key string, defs ...bool) bool {
ck, err := c.req.Cookie(key)
if err != nil {
if len(defs) > 0 {
return defs[0]
}
return false
}
s := parseSecureCookie(c.secret, ck.Value)
v, err := strconv.ParseBool(s)
if len(defs) > 0 && err != nil {
return defs[0]
}
return v
}
func secCookieValue(secret string, vb []byte) string {
timestamp := strconv.FormatInt(time.Now().Unix(), 10)
sig := getCookieSig(secret, vb, timestamp)
return strings.Join([]string{string(vb), timestamp, sig}, "|")
}
func NewSecureCookie(secret, name string, val string, age ...int64) *http.Cookie {
var buf bytes.Buffer
encoder := base64.NewEncoder(base64.StdEncoding, &buf)
encoder.Write([]byte(val))
encoder.Close()
cookie := secCookieValue(secret, buf.Bytes())
return NewCookie(name, cookie, age...)
}
func (c *secureCookies) Expire(key string, expire time.Time) {
ck := c.Get(key)
if ck != nil {
ck.Expires = expire
ck.Value = secCookieValue(c.secret, []byte(ck.Value))
c.Set(ck)
}
}
func (c *secureCookies) Del(key string) {
c.Expire(key, time.Date(1900, 1, 1, 0, 0, 0, 0, time.Local))
}
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/sunstone1997/tango.git
git@gitee.com:sunstone1997/tango.git
sunstone1997
tango
tango
master

搜索帮助