国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁(yè) > 編程 > Golang > 正文

詳解Golang實(shí)現(xiàn)http重定向https的方式

2020-04-01 18:54:13
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

以前寫(xiě)代碼時(shí),都是直接將程序綁定到唯一端口提供http/https服務(wù),在外層通過(guò)反向代理(nginx/caddy)來(lái)實(shí)現(xiàn)http和https的切換。隨著上線后的服務(wù)越來(lái)越多,有一些服務(wù)無(wú)法直接通過(guò)反向代理來(lái)提供這種重定向,只能依靠代碼自己實(shí)現(xiàn)。所以簡(jiǎn)要記錄一下如何在代碼中實(shí)現(xiàn)http到https的重定向。

分析

無(wú)論是反向代理還是代碼自己實(shí)現(xiàn),問(wèn)題的本質(zhì)都是判斷請(qǐng)求是否是https請(qǐng)求。 如果是則直接處理,如果不是,則修改請(qǐng)求中的url地址,同時(shí)返回客戶端一個(gè)重定向狀態(tài)碼(301/302/303/307)。但如果仔細(xì)分析的話,會(huì)衍生出另外的問(wèn)題,返回哪個(gè)重定向碼是合理的?

這個(gè)問(wèn)題展開(kāi)討論,估計(jì)要寫(xiě)滿滿一大頁(yè),可能還得不出結(jié)論。 因此這里就不糾結(jié)到底返回哪個(gè)了,我使用的是307.

實(shí)現(xiàn)

如何我們從問(wèn)題出現(xiàn)的場(chǎng)景開(kāi)始分析,基本可以得出一個(gè)結(jié)論: 在需要轉(zhuǎn)換的場(chǎng)景中,都是用戶習(xí)慣性的首先發(fā)出了http請(qǐng)求,然后服務(wù)器才需要返回一個(gè)https的重定向。 因此實(shí)現(xiàn)的第一步就是創(chuàng)建一個(gè)監(jiān)聽(tīng)http請(qǐng)求的端口:

go http.ListenAndServe(":8000", http.HandlerFunc(redirect))

8000端口專門(mén)用來(lái)監(jiān)聽(tīng)http請(qǐng)求,不能阻塞https主流程,因此單獨(dú)扔給一個(gè)協(xié)程來(lái)處理。 redirect用來(lái)實(shí)現(xiàn)重定向:

func redirect(w http.ResponseWriter, req *http.Request) {   _host := strings.Split(req.Host, ":")  _host[1] = "8443"  target := "https://" + strings.Join(_host, ":") + req.URL.Path  if len(req.URL.RawQuery) > 0 {    target += "?" + req.URL.RawQuery  }  http.Redirect(w, req, target, http.StatusTemporaryRedirect)}

8443是https監(jiān)聽(tīng)的端口。 如果監(jiān)聽(tīng)默認(rèn)端口443,那么就可加可不加。 最后調(diào)用sdk中的Redirect函數(shù)封裝Response。

處理完重定向之后,再處理https就變得很容易了:

router := mux.NewRouter()   router.Path("/").HandlerFunc(handleHttps)  c := cors.New(cors.Options{    AllowedOrigins:  []string{"*.devexp.cn"},    AllowedMethods:  []string{"HEAD", "GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS"},    AllowedHeaders:  []string{"*"},    AllowCredentials: true,    Debug:      false,    AllowOriginFunc: func(origin string) bool {      return true    },  })  handler := c.Handler(router)  logrus.Fatal(http.ListenAndServeTLS(":8443", "cert.crt", "cert.key", handler))

完整代碼如下:

package mainimport (   "github.com/gorilla/mux"  "github.com/rs/cors"  "github.com/sirupsen/logrus"  "net/http"  "encoding/json"  "log"  "strings")func main() {   go http.ListenAndServe(":8000", http.HandlerFunc(redirect))  router := mux.NewRouter()  router.Path("/").HandlerFunc(handleHttps)  c := cors.New(cors.Options{    AllowedOrigins:  []string{"*.devexp.cn"},    AllowedMethods:  []string{"HEAD", "GET", "POST", "PUT", "PATCH", "DELETE", "OPTIONS"},    AllowedHeaders:  []string{"*"},    AllowCredentials: true,    Debug:      false,    AllowOriginFunc: func(origin string) bool {      return true    },  })  handler := c.Handler(router)  logrus.Fatal(http.ListenAndServeTLS(":8443", "cert.crt", "cert.key", handler))}func redirect(w http.ResponseWriter, req *http.Request) {   _host := strings.Split(req.Host, ":")  _host[1] = "8443"  // remove/add not default ports from req.Host  target := "https://" + strings.Join(_host, ":") + req.URL.Path  if len(req.URL.RawQuery) > 0 {    target += "?" + req.URL.RawQuery  }  log.Printf("redirect to: %s", target)  http.Redirect(w, req, target,    // see @andreiavrammsd comment: often 307 > 301    http.StatusTemporaryRedirect)}func handleHttps(w http.ResponseWriter, r *http.Request) {   json.NewEncoder(w).Encode(struct {    Name string    Age  int    Https bool  }{    "lala",    11,    true,  })}

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持VEVB武林網(wǎng)。


發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 饶河县| 浏阳市| 陵川县| 郧西县| 孟津县| 阳曲县| 芒康县| 噶尔县| 安化县| 谢通门县| 河池市| 桃江县| 金门县| 新乡县| 荣成市| 阿图什市| 青铜峡市| 泰顺县| 玉龙| 和龙市| 达拉特旗| 阳朔县| 上杭县| 广灵县| 内黄县| 会同县| 哈密市| 工布江达县| 当雄县| 江津市| 榆林市| 贵南县| 明溪县| 洮南市| 武邑县| 威宁| 光山县| 民乐县| 舒兰市| 三门县| 东方市|