您好,登錄后才能下訂單哦!
暫且放下你的編程語言來瞻仰下我所見過的最棒的標(biāo)準(zhǔn)庫。
為項(xiàng)目選擇編程語言和挑選你最愛的球隊(duì)不一樣。應(yīng)該從實(shí)用主義出發(fā),根據(jù)特定的工作選擇合適的工具。
在這篇文章中我會(huì)告訴你從何時(shí)開始并且為什么我認(rèn)為 Go 語言如此閃耀,具體來說是它的標(biāo)準(zhǔn)庫對(duì)于基本的網(wǎng)絡(luò)編程來說顯得非常穩(wěn)固。更具體一點(diǎn),我們將要編寫一個(gè)反向代理程序。
Go 為此提供了很多,但真正支撐起它的在于這些低級(jí)的網(wǎng)絡(luò)管道任務(wù),沒有更好的語言了。
反向代理是什么? 有個(gè)很棒的說法是流量轉(zhuǎn)發(fā) 。我獲取到客戶端來的請(qǐng)求,將它發(fā)往另一個(gè)服務(wù)器,從服務(wù)器獲取到響應(yīng)再回給原先的客戶端。反向的意義簡(jiǎn)單來說在于這個(gè)代理自身決定了何時(shí)將流量發(fā)往何處。
為什么這很有用?因?yàn)榉聪虼淼母拍钍侨绱撕?jiǎn)單以至于它可以被應(yīng)用于許多不同的場(chǎng)景:負(fù)載均衡,A/B 測(cè)試,高速緩存,驗(yàn)證等等。
當(dāng)讀完這篇文章之后,你會(huì)學(xué)到:
我們的反向代理項(xiàng)目
我們來實(shí)際寫一下項(xiàng)目。我們需要一個(gè) Web 服務(wù)器能夠提供以下功能:
準(zhǔn)備工作
環(huán)境配置
我們要做的第一件事是將我們的配置信息寫入環(huán)境變量,如此就可以使用它們而不必寫死在我們的源代碼中。
我發(fā)現(xiàn)最好的方式是創(chuàng)建一個(gè)包含所需環(huán)境變量的 .env
文件。
以下就是我為特定項(xiàng)目編寫的文件內(nèi)容:
export PORT=1330 export A_CONDITION_URL="http://localhost:1331" export B_CONDITION_URL="http://localhost:1332" export DEFAULT_CONDITION_URL=http://localhost:1333
這是我從 12 Factor App 項(xiàng)目中獲得的技巧。
保存完 .env
文件之后就可以運(yùn)行:
source .env
在任何時(shí)候都可以運(yùn)行該指令來將配置加載進(jìn)環(huán)境變量。
項(xiàng)目基礎(chǔ)工作
接著我們創(chuàng)建 main.go
文件做如下事情:
PORT
, A_CONDITION_URL
, B_CONDITION_URL
和 DEFAULT_CONDITION_URL
變量通過日志打印到控制臺(tái)。/
路徑上監(jiān)聽請(qǐng)求:package main import ( "bytes" "encoding/json" "io/ioutil" "log" "net/http" "net/http/httputil" "net/url" "os" "strings" ) // Get env var or default func getEnv(key, fallback string) string { if value, ok := os.LookupEnv(key); ok { return value } return fallback } // Get the port to listen on func getListenAddress() string { port := getEnv("PORT", "1338") return ":" + port } // Log the env variables required for a reverse proxy func logSetup() { a_condtion_url := os.Getenv("A_CONDITION_URL") b_condtion_url := os.Getenv("B_CONDITION_URL") default_condtion_url := os.Getenv("DEFAULT_CONDITION_URL") log.Printf("Server will run on: %s\n", getListenAddress()) log.Printf("Redirecting to A url: %s\n", a_condtion_url) log.Printf("Redirecting to B url: %s\n", b_condtion_url) log.Printf("Redirecting to Default url: %s\n", default_condtion_url) } // Given a request send it to the appropriate url func handleRequestAndRedirect(res http.ResponseWriter, req *http.Request) { // We will get to this... } func main() { // Log setup values logSetup() // start server http.HandleFunc("/", handleRequestAndRedirect) if err := http.ListenAndServe(getListenAddress(), nil); err != nil { panic(err) } }
現(xiàn)在你就可以運(yùn)行代碼了。
解析請(qǐng)求體
有了項(xiàng)目的基本骨架之后,我們需要添加邏輯來處理解析請(qǐng)求的請(qǐng)求體部分。更新 handleRequestAndRedirect
函數(shù)來從請(qǐng)求體中解析出 proxy_condition
字段。
type requestPayloadStruct struct { ProxyCondition string `json:"proxy_condition"` } // Get a json decoder for a given requests body func requestBodyDecoder(request *http.Request) *json.Decoder { // Read body to buffer body, err := ioutil.ReadAll(request.Body) if err != nil { log.Printf("Error reading body: %v", err) panic(err) } // Because go lang is a pain in the ass if you read the body then any susequent calls // are unable to read the body again.... request.Body = ioutil.NopCloser(bytes.NewBuffer(body)) return json.NewDecoder(ioutil.NopCloser(bytes.NewBuffer(body))) } // Parse the requests body func parseRequestBody(request *http.Request) requestPayloadStruct { decoder := requestBodyDecoder(request) var requestPayload requestPayloadStruct err := decoder.Decode(&requestPayload) if err != nil { panic(err) } return requestPayload } // Given a request send it to the appropriate url func handleRequestAndRedirect(res http.ResponseWriter, req *http.Request) { requestPayload := parseRequestBody(req) // ... more to come }
通過 proxy_condition 判斷將流量發(fā)往何處
現(xiàn)在我們從請(qǐng)求中取得了 proxy_condition
的值,可以根據(jù)它來判斷我們要反向代理到何處。記住上文我們提到的三種情形:
proxy_condition
值為 A
,我們將流量發(fā)送到 A_CONDITION_URL
proxy_condition
值為 B
,我們將流量發(fā)送到 B_CONDITION_URL
DEFAULT_CONDITION_URL
// Log the typeform payload and redirect url func logRequestPayload(requestionPayload requestPayloadStruct, proxyUrl string) { log.Printf("proxy_condition: %s, proxy_url: %s\n", requestionPayload.ProxyCondition, proxyUrl) } // Get the url for a given proxy condition func getProxyUrl(proxyConditionRaw string) string { proxyCondition := strings.ToUpper(proxyConditionRaw) a_condtion_url := os.Getenv("A_CONDITION_URL") b_condtion_url := os.Getenv("B_CONDITION_URL") default_condtion_url := os.Getenv("DEFAULT_CONDITION_URL") if proxyCondition == "A" { return a_condtion_url } if proxyCondition == "B" { return b_condtion_url } return default_condtion_url } // Given a request send it to the appropriate url func handleRequestAndRedirect(res http.ResponseWriter, req *http.Request) { requestPayload := parseRequestBody(req) url := getProxyUrl(requestPayload.ProxyCondition) logRequestPayload(requestPayload, url) // more still to come... }
反向代理到 URL
最終我們來到了實(shí)際的反向代理部分。在如此多的語言中要編寫一個(gè)反向代理需要考慮很多東西,寫大段的代碼。或者至少引入一個(gè)復(fù)雜的外部庫。
然而 Go 的標(biāo)準(zhǔn)庫使得創(chuàng)建一個(gè)反向代理非常簡(jiǎn)單以至于你都不敢相信。下面就是你所需要的最關(guān)鍵的一行代碼:
httputil.NewSingleHostReverseProxy(url).ServeHTTP(res, req)
注意下面代碼中我們做了些許修改來讓它能完整地支持 SSL 重定向(雖然不是必須的)。
// Serve a reverse proxy for a given url func serveReverseProxy(target string, res http.ResponseWriter, req *http.Request) { // parse the url url, _ := url.Parse(target) // create the reverse proxy proxy := httputil.NewSingleHostReverseProxy(url) // Update the headers to allow for SSL redirection req.URL.Host = url.Host req.URL.Scheme = url.Scheme req.Header.Set("X-Forwarded-Host", req.Header.Get("Host")) req.Host = url.Host // Note that ServeHttp is non blocking and uses a go routine under the hood proxy.ServeHTTP(res, req) } // Given a request send it to the appropriate url func handleRequestAndRedirect(res http.ResponseWriter, req *http.Request) { requestPayload := parseRequestBody(req) url := getProxyUrl(requestPayload.ProxyCondition) logRequestPayload(requestPayload, url) serveReverseProxy(url, res, req) }
全部啟動(dòng)
好了,現(xiàn)在啟動(dòng)我們的反向代理程序讓其監(jiān)聽 1330
端口。讓其他的 3 個(gè)簡(jiǎn)單的服務(wù)分別監(jiān)聽 1331–1333
端口(在各自的終端中)。
這些服務(wù)都啟動(dòng)之后,我們就可以在另一個(gè)終端中像下面這樣開始發(fā)送帶有 JSON 體的請(qǐng)求了:
curl --request GET \ --url http://localhost:1330/ \ --header 'content-type: application/json' \ --data '{ "proxy_condition": "a" }'
如果你在找一個(gè)好用的 HTTP 請(qǐng)求客戶端,我極力推薦 Insomnia 。
然后我們就會(huì)看到我們的反向代理將流量轉(zhuǎn)發(fā)給了我們根據(jù) proxy_condition
字段配置的 3 臺(tái)服務(wù)中的其中一臺(tái)。
總結(jié)
Go 為此提供了很多,但真正支撐起它的在于這些低級(jí)的網(wǎng)絡(luò)管道任務(wù),沒有更好的語言了。我們寫的這個(gè)程序簡(jiǎn)單,高性能,可靠并且隨時(shí)可用于生產(chǎn)環(huán)境。
我能看到在以后我會(huì)經(jīng)常使用 Go 來編寫簡(jiǎn)單的服務(wù)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
代碼是開源的,你可以在 Github 上找到。 :heart: 在 Twitter 上我只聊關(guān)于編程和遠(yuǎn)程工作相關(guān)的東西。如果關(guān)注我,你不會(huì)后悔的。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。