会话的最佳实践(大猩猩/会话)

时间:2014-09-29 18:23:01

标签: session go

在开始使用golang中的会话之前,我需要回答一些问题

会话示例

import "github.com/gorilla/sessions"

var store = sessions.NewCookieStore([]byte("33446a9dcf9ea060a0a6532b166da32f304af0de"))

func Handler(w http.ResponseWriter, r *http.Request){
    session, _ := store.Get(r, "session-name")

    session.Values["foo"] = "bar"
    session.Values[42] = 43
    session.Save(r, w)

    fmt.Fprint(w, "Hello world :)")
}

func main(){
    store.Options = &sessions.Options{
        Domain:     "localhost",
        Path:       "/",
        MaxAge:     60 * 15,
        Secure:     false,
        HttpOnly:   true,
    }
}

Q1:

是否可以在同一个域中使用不同的名称添加多个会话?

session1, _ := store.Get(r, "session-name-1")
session2, _ := store.Get(r, "session-name-2")

您何时需要在同一个域上进行多个会话?

Q2:

从会话中获取变量的最佳做法是什么? my_session_var = session.Values [“foo”]

Q3:

如何检查会话是否正确保存?如果您同时访问map同时设置和获取变量?

更新

package main

import (
    "github.com/gorilla/sessions"
)

var (
    store = sessions.NewCookieStore([]byte("33446a9dcf9ea060a0a6532b166da32f304af0de"))
)

type handler func(w http.ResponseWriter, r *http.Request, s *sessions.Session)

func (h handler) ServeHTTP(w http.ResponseWriter, r *http.Request){
    session, _ := store.Get(r, "session-name")

    h(w, r, session)
}

func Handler_404(w http.ResponseWriter, r *http.Request, s *sessions.Session){
    fmt.Fprint(w, "Oops, something went wrong!")
}

错误

# command-line-arguments
.\mux.go:101: cannot convert Handler_404 (type func(http.ResponseWriter, *http.Request, *sessions.Session)) to type http.HandlerFunc

1 个答案:

答案 0 :(得分:5)

文章“BASIC EXTENSION OF GO’S HTTP HANDLERS”(Simon Whitehead)显示了定义会话的位置和时间的示例。
而不是在Handler本身中执行此操作,并且在定义其他处理程序时必须复制大量代码。

使用named type,您可以定义所需的Handler

type handler func(w http.ResponseWriter, r *http.Request, db *mgo.Database)

(在您的情况下,它将是一个大猩猩会话而不是mgo会话或数据库)

init() function可以处理会话创建(这里是mgo会话,但其他框架会话的想法是一样的)

func init() {
    session, err = mgo.Dial("localhost")

    if err != nil {
        log.Println(err)
    }
}

你可以确保这个函数类型('handler')确实尊重ServeHTTP()函数,负责:

  • 会话管理(克隆/关闭)
  • 调用您的实际处理程序(可以包含比wr更多的参数)

    func (h handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
        s := session.Clone()
        defer s.Close()
    
        h(w, r, s.DB("example"))
    }
    

然后,您可以定义实际的Handler(再次,超过wr):

func myHandler(w http.ResponseWriter, r *http.Request, db *mgo.Database) {
    var users []user

    db.C("users").Find(nil).All(&users)

    for _, user := range users {
        fmt.Fprintf(w, "%s is %d years old", user.Name, user.Age)
    }
}

您可以在服务器中使用该处理程序:

func main() {
    mux := http.NewServeMux()
    mux.Handle("/", handler(myHandler))
    http.ListenAndServe(":8080", mux)
}

我们的想法是将main()中的“管道”限制在最低限度,同时使Handler包含更多参数(包括您的会话)。
这允许您使用非常少的管道使用不同的Handlers,仅保留main()来声明不同的路径(而不是初始化会话和处理程序)


更新2019:在另一个相关背景中,另请参阅“How to handle sessions”。