http.Handler 在golang中没有被调用
http.Handler is not called in golang
我有以下中间件,首先将 gorilla/context
中的 currentUser
设置为从数据库获取的当前用户,第二个检查 currentUser
是否存在,否则重定向:
package main
import (
"database/sql"
"github.com/gorilla/context"
"log"
"net/http"
"server/helpers"
)
func withCurrentUser(db *sql.DB, next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
userId := helpers.GetCurrentUserId(db, r)
if userId == nil {
next.ServeHTTP(w, r)
return
}
row := db.QueryRow("SELECT username FROM User WHERE id=?", userId)
var username string
switch err := row.Scan(&username); err {
case sql.ErrNoRows:
next.ServeHTTP(w, r)
return
case nil:
user := helpers.User{UserId: *userId, LoggedIn: true, Username: username}
context.Set(r, "currentUser", user)
default:
log.Fatal(err)
}
next.ServeHTTP(w, r)
})
}
func loginRequired(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
user, ok := context.Get(r, "currentUser").(helpers.User)
log.Println(user, ok)
if !ok {
http.Redirect(w, r, "/login", 301)
return
}
next.ServeHTTP(w, r)
})
}
然后当我注册一个需要经过身份验证的用户的路由时,我会执行以下操作:
router.Handle("/create_post",
withCurrentUser(db, loginRequired(http.HandlerFunc(createPostGet))),
).Methods(http.MethodGet)
其中 createPostGet
:
func createPostGet(w http.ResponseWriter, r *http.Request) {
tmpl := template.Must(template.ParseFiles("templates/createPost.html"))
user := context.Get(r, "currentUser").(helpers.User)
_ = tmpl.Execute(w, helpers.FormStruct{CurrentUser: user})
}
我的问题是:即使用户已通过身份验证并且上下文已正确填充,此路由始终会重定向到登录。我尝试在 loginRequired
内设置断点并添加 log.Println
(正如您在代码中看到的那样),并且该函数似乎甚至没有被调用(没有断点停止,也没有日志输出)。
发生了什么以及如何确保每次调用 loginRequired
并正确检查上下文?
UPD: 它似乎没有持续存在,我重新编译了几次该应用程序,现在它可以运行了。无论如何,这种行为的原因可能是什么(我肯定我第一次保存了所有内容)。
UPD 2: 我发现问题与浏览器缓存有关,但我仍然不知道为什么会这样。当我禁用浏览器缓存时,一切正常并且每次都会调用函数,但是当启用浏览器缓存时,根本不会调用该函数。有什么想法吗?
天哪,这太愚蠢了。我在 loginRequired
中使用了 301 重定向代码,因此重定向是永久性的,浏览器甚至没有发出请求。
我有以下中间件,首先将 gorilla/context
中的 currentUser
设置为从数据库获取的当前用户,第二个检查 currentUser
是否存在,否则重定向:
package main
import (
"database/sql"
"github.com/gorilla/context"
"log"
"net/http"
"server/helpers"
)
func withCurrentUser(db *sql.DB, next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
userId := helpers.GetCurrentUserId(db, r)
if userId == nil {
next.ServeHTTP(w, r)
return
}
row := db.QueryRow("SELECT username FROM User WHERE id=?", userId)
var username string
switch err := row.Scan(&username); err {
case sql.ErrNoRows:
next.ServeHTTP(w, r)
return
case nil:
user := helpers.User{UserId: *userId, LoggedIn: true, Username: username}
context.Set(r, "currentUser", user)
default:
log.Fatal(err)
}
next.ServeHTTP(w, r)
})
}
func loginRequired(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
user, ok := context.Get(r, "currentUser").(helpers.User)
log.Println(user, ok)
if !ok {
http.Redirect(w, r, "/login", 301)
return
}
next.ServeHTTP(w, r)
})
}
然后当我注册一个需要经过身份验证的用户的路由时,我会执行以下操作:
router.Handle("/create_post",
withCurrentUser(db, loginRequired(http.HandlerFunc(createPostGet))),
).Methods(http.MethodGet)
其中 createPostGet
:
func createPostGet(w http.ResponseWriter, r *http.Request) {
tmpl := template.Must(template.ParseFiles("templates/createPost.html"))
user := context.Get(r, "currentUser").(helpers.User)
_ = tmpl.Execute(w, helpers.FormStruct{CurrentUser: user})
}
我的问题是:即使用户已通过身份验证并且上下文已正确填充,此路由始终会重定向到登录。我尝试在 loginRequired
内设置断点并添加 log.Println
(正如您在代码中看到的那样),并且该函数似乎甚至没有被调用(没有断点停止,也没有日志输出)。
发生了什么以及如何确保每次调用 loginRequired
并正确检查上下文?
UPD: 它似乎没有持续存在,我重新编译了几次该应用程序,现在它可以运行了。无论如何,这种行为的原因可能是什么(我肯定我第一次保存了所有内容)。
UPD 2: 我发现问题与浏览器缓存有关,但我仍然不知道为什么会这样。当我禁用浏览器缓存时,一切正常并且每次都会调用函数,但是当启用浏览器缓存时,根本不会调用该函数。有什么想法吗?
天哪,这太愚蠢了。我在 loginRequired
中使用了 301 重定向代码,因此重定向是永久性的,浏览器甚至没有发出请求。