在Go語言的Web框架go-chi中,為子路由設置不同的中間件是一種常見的需求。中間件是一種處理HTTP請求和響應的函數,它可以在請求到達目標處理器之前和之后執行一些額外的操作。通過為不同的子路由設置不同的中間件,我們可以根據業務需求對每個子路由進行個性化的處理。在本文中,我們將介紹如何在go-chi中靈活地為子路由設置不同的中間件,以實現更加靈活的路由處理。
問題內容
我有一個可以通過配置停用的代理服務器,使用 mustBeSecure
。我希望這個代理在某個子域中表現不同:“/application/health”,使其始終不安全。到目前為止我嘗試過的所有更改都失敗了。有沒有辦法為“/application/health”配置不同的組,該組仍然使用代理但從不需要身份驗證?
router := chi.NewRouter() router.Route("/", func(r chi.Router) { r.Use(chimw.Recoverer) router.Use(hlog.NewHandler(log.Logger)) if mustBeSecure() { r.Use(keycloak.MustStandardKeycloakAuth("url")) } setProxy(r) r.Group(func(r chi.Router) { r.Get("/health", handleHealth()) }) }) return http.ListenAndServe("0.0.0.0", router)
登錄后復制
根據要求,這里是 setProxy() 的示例
func setProxy(r chi.Router) { r.Route("/application", func(r chi.Router) { r.Use(func(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { ctx := context.WithValue(r.Context(), "target", "http:locahost:9999") r = r.WithContext(ctx) next.ServeHTTP(w, r) }) }) r.HandleFunc("/*", httputil.ReverseProxy{}.ServeHTTP) }) }
登錄后復制
解決方法
我相信你正在嘗試這個。
對所有 /application
路由使用路由器
聲明 GET /health
為所有其他路由創建一個組(在 /application
中)
應用條件中間件
為了便于理解,我稍微簡化了代碼。您仍然可以使用 setProxy
函數。
func main() { proxy, err := NewProxy("http://localhost:4422") if err != nil { panic(err) } router := chi.NewRouter() router.Get("/health", handleHealth) router.Route("/application", func(r chi.Router) { r.Get("/health", ProxyRequestHandler(proxy)) r.Group(func(r chi.Router) { if mustBeSecure() { r.Use(keycloak.MustStandardKeycloakAuth("url")) } r.HandleFunc("/*", ProxyRequestHandler(proxy)) }) }) http.ListenAndServe("0.0.0.0:4411", router) } func NewProxy(targetHost string) (*httputil.ReverseProxy, error) { targetUrl, err := url.Parse(targetHost) if err != nil { return nil, err } return httputil.NewSingleHostReverseProxy(targetUrl), nil } func ProxyRequestHandler(proxy *httputil.ReverseProxy) func(http.ResponseWriter, *http.Request) { return func(w http.ResponseWriter, r *http.Request) { proxy.ServeHTTP(w, r) } }
登錄后復制