校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃

主頁 > 知識庫 > 1行Go代碼實現反向代理的示例

1行Go代碼實現反向代理的示例

熱門標簽:廣州呼叫中心外呼系統 南通如皋申請開通400電話 地圖標注的汽車標 江西轉化率高的羿智云外呼系統 高德地圖標注口訣 浙江高速公路地圖標注 中國地圖標注省會高清 西部云谷一期地圖標注 學海導航地圖標注

暫且放下你的編程語言來瞻仰下我所見過的最棒的標準庫。

為項目選擇編程語言和挑選你最愛的球隊不一樣。應該從實用主義出發,根據特定的工作選擇合適的工具。

在這篇文章中我會告訴你從何時開始并且為什么我認為 Go 語言如此閃耀,具體來說是它的標準庫對于基本的網絡編程來說顯得非常穩固。更具體一點,我們將要編寫一個反向代理程序。

Go 為此提供了很多,但真正支撐起它的在于這些低級的網絡管道任務,沒有更好的語言了。

反向代理是什么? 有個很棒的說法是流量轉發 。我獲取到客戶端來的請求,將它發往另一個服務器,從服務器獲取到響應再回給原先的客戶端。反向的意義簡單來說在于這個代理自身決定了何時將流量發往何處。

為什么這很有用?因為反向代理的概念是如此簡單以至于它可以被應用于許多不同的場景:負載均衡,A/B 測試,高速緩存,驗證等等。

當讀完這篇文章之后,你會學到:

  • 如何響應 HTTP 請求
  •  如何解析請求體
  • 如何通過反向代理將流量轉發到另一臺服務器

 我們的反向代理項目

我們來實際寫一下項目。我們需要一個 Web 服務器能夠提供以下功能:

  • 獲取到請求
  • 讀取請求體,特別是 proxy_condition 字段
  • 如果代理域為 A,則轉發到 URL 1
  • 如果代理域為 B,則轉發到 URL 2
  • 如果代理域都不是以上,則轉發到默認的 URL

準備工作

  •  Go 語言環境。
  • http-server 用來創建簡單的服務。

環境配置

我們要做的第一件事是將我們的配置信息寫入環境變量,如此就可以使用它們而不必寫死在我們的源代碼中。

我發現最好的方式是創建一個包含所需環境變量的 .env 文件。

以下就是我為特定項目編寫的文件內容:

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 項目中獲得的技巧。

保存完 .env 文件之后就可以運行:

source .env

在任何時候都可以運行該指令來將配置加載進環境變量。

項目基礎工作

接著我們創建 main.go 文件做如下事情:

  1. PORT , A_CONDITION_URLB_CONDITION_URLDEFAULT_CONDITION_URL 變量通過日志打印到控制臺。
  2. / 路徑上監聽請求:
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)
 }
}

現在你就可以運行代碼了。

解析請求體

有了項目的基本骨架之后,我們需要添加邏輯來處理解析請求的請求體部分。更新 handleRequestAndRedirect 函數來從請求體中解析出 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 判斷將流量發往何處

現在我們從請求中取得了 proxy_condition 的值,可以根據它來判斷我們要反向代理到何處。記住上文我們提到的三種情形:

  • 如果 proxy_condition 值為 A ,我們將流量發送到 A_CONDITION_URL
  • 如果 proxy_condition 值為 B ,我們將流量發送到 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

最終我們來到了實際的反向代理部分。在如此多的語言中要編寫一個反向代理需要考慮很多東西,寫大段的代碼?;蛘咧辽僖胍粋€復雜的外部庫。

然而 Go 的標準庫使得創建一個反向代理非常簡單以至于你都不敢相信。下面就是你所需要的最關鍵的一行代碼:

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)
}

全部啟動

好了,現在啟動我們的反向代理程序讓其監聽 1330 端口。讓其他的 3 個簡單的服務分別監聽 1331–1333 端口(在各自的終端中)。

  1. source .env go install $GOPATH/bin/reverse-proxy-demo
  2. http-server -p 1331
  3. http-server -p 1332
  4. http-server -p 1333

這些服務都啟動之后,我們就可以在另一個終端中像下面這樣開始發送帶有 JSON 體的請求了:

curl --request GET \

 --url http://localhost:1330/ \

 --header 'content-type: application/json' \

 --data '{
 "proxy_condition": "a"
 }'

如果你在找一個好用的 HTTP 請求客戶端,我極力推薦 Insomnia 。

然后我們就會看到我們的反向代理將流量轉發給了我們根據 proxy_condition 字段配置的 3 臺服務中的其中一臺。

總結

Go 為此提供了很多,但真正支撐起它的在于這些低級的網絡管道任務,沒有更好的語言了。我們寫的這個程序簡單,高性能,可靠并且隨時可用于生產環境。

我能看到在以后我會經常使用 Go 來編寫簡單的服務。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

‍ 代碼是開源的,你可以在 Github 上找到。 :heart: 在 Twitter 上我只聊關于編程和遠程工作相關的東西。如果關注我,你不會后悔的。

您可能感興趣的文章:
  • go語言實現簡單http服務的方法
  • go語言實現一個簡單的http客戶端抓取遠程url的方法
  • go語言實現一個最簡單的http文件服務器實例
  • 一個簡單的Golang實現的HTTP Proxy方法

標簽:曲靖 東營 德宏 保定 貴州 吐魯番 常州 許昌

巨人網絡通訊聲明:本文標題《1行Go代碼實現反向代理的示例》,本文關鍵詞  1行,代碼,實現,反向,代理,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《1行Go代碼實現反向代理的示例》相關的同類信息!
  • 本頁收集關于1行Go代碼實現反向代理的示例的相關信息資訊供網民參考!
  • 推薦文章
    校园春色亚洲色图_亚洲视频分类_中文字幕精品一区二区精品_麻豆一区区三区四区产品精品蜜桃
    久久精品二区亚洲w码| 色综合色综合色综合色综合色综合| 国产xxx精品视频大全| 日本韩国精品在线| 久久久不卡网国产精品二区| 亚洲成人一二三| 色综合天天综合给合国产| 精品处破学生在线二十三| 天堂一区二区在线| 一本一道久久a久久精品综合蜜臀| 欧美变态tickling挠脚心| 亚洲国产另类精品专区| 一本到一区二区三区| 欧美韩国日本不卡| 国产高清久久久久| 精品国免费一区二区三区| 日本不卡一区二区| 欧美日本国产视频| 午夜精品久久久久影视| 91激情在线视频| 一区二区三国产精华液| 99久久婷婷国产| 国产欧美日韩精品a在线观看| 久久www免费人成看片高清| 欧美精品乱码久久久久久| 亚洲午夜在线观看视频在线| 在线观看日韩高清av| 一区二区三区在线视频观看| 色乱码一区二区三区88| 亚洲精品va在线观看| 91热门视频在线观看| 成人免费一区二区三区视频| 不卡一区在线观看| 亚洲裸体在线观看| 91黄色激情网站| 亚洲成人精品一区| 日韩欧美在线一区二区三区| 蜜臀精品一区二区三区在线观看| 欧美一区二区女人| 韩国三级在线一区| 国产午夜精品理论片a级大结局| 国产精品一区在线| 综合久久一区二区三区| 欧美在线一区二区三区| 日产国产欧美视频一区精品| 精品av综合导航| 成人午夜在线视频| 亚洲一区二区精品视频| 日韩精品一区在线观看| 国产 欧美在线| 亚洲午夜久久久| 日韩小视频在线观看专区| 国产一区 二区| 亚洲蜜桃精久久久久久久| 91精品国产综合久久久蜜臀粉嫩| 精品一区二区三区免费观看| 国产精品乱码一区二三区小蝌蚪| 91天堂素人约啪| 秋霞国产午夜精品免费视频| 久久久精品国产免费观看同学| fc2成人免费人成在线观看播放 | 美腿丝袜亚洲综合| 久久亚洲综合av| 在线观看日产精品| 国产一区二区三区香蕉| 亚洲精品美国一| 欧美成人性福生活免费看| 色婷婷av一区二区三区gif| 美女尤物国产一区| 一区在线播放视频| 欧美大黄免费观看| 色哟哟一区二区在线观看| 麻豆91精品视频| 亚洲美腿欧美偷拍| 国产亚洲精品精华液| 7777女厕盗摄久久久| 成人中文字幕电影| 久久国产精品99精品国产| 亚洲欧美色一区| 久久久99久久| 日韩欧美的一区| 成人福利电影精品一区二区在线观看| 午夜免费欧美电影| 亚洲欧美日韩电影| 国产午夜亚洲精品理论片色戒| 欧美巨大另类极品videosbest | 国产精品久久国产精麻豆99网站| 91精品在线免费观看| 91在线免费播放| 丁香激情综合五月| 国产自产2019最新不卡| 丝袜亚洲精品中文字幕一区| 亚洲美女偷拍久久| 国产精品成人免费| 国产精品五月天| 国产日韩欧美电影| 久久免费的精品国产v∧| 制服丝袜亚洲色图| 精品视频免费在线| 91国偷自产一区二区三区成为亚洲经典 | 国模少妇一区二区三区| 天堂成人免费av电影一区| 亚洲精品免费看| 亚洲精品免费在线播放| 亚洲精品乱码久久久久久日本蜜臀| 国产欧美一区二区精品性色超碰| 精品福利一区二区三区免费视频| 欧美精品久久一区二区三区| 欧美精品一二三区| 欧美日本在线一区| 91精品国产综合久久精品app| 精品视频一区三区九区| 欧美婷婷六月丁香综合色| 欧美在线|欧美| 欧美性大战久久久久久久| 欧美少妇一区二区| 欧美日韩成人在线| 欧美一区二区免费观在线| 26uuu国产电影一区二区| 精品国产免费久久| 国产亚洲成aⅴ人片在线观看| 久久久国产精品不卡| 成人免费在线视频观看| 一区二区三区加勒比av| 亚洲成av人片在www色猫咪| 午夜免费久久看| 久久er精品视频| 国产1区2区3区精品美女| 91老师片黄在线观看| 欧美午夜电影在线播放| 717成人午夜免费福利电影| 日韩欧美国产一二三区| 国产欧美久久久精品影院| 亚洲男同性恋视频| 日本一区中文字幕| 国产精品一区二区三区乱码| 99热精品国产| 日韩一级欧美一级| 国产精品欧美一区喷水| 亚洲一区免费在线观看| 精品中文字幕一区二区小辣椒| 国产成人自拍网| 欧美日韩另类一区| 26uuu国产日韩综合| 一区二区三区不卡视频在线观看| 蜜臀av性久久久久av蜜臀妖精| 国产成人精品免费| 欧美综合天天夜夜久久| 欧美sm美女调教| 一区二区三区欧美在线观看| 久久99精品久久久| 日本久久电影网| 久久精品一区蜜桃臀影院| 亚洲伦在线观看| 国产一区视频导航| 欧美三区在线视频| 亚洲国产精品激情在线观看| 丝袜亚洲另类欧美综合| 99综合电影在线视频| 精品日韩一区二区三区免费视频| 亚洲麻豆国产自偷在线| 国产美女av一区二区三区| 色88888久久久久久影院按摩| 久久视频一区二区| 婷婷一区二区三区| 99久久精品免费| 久久久久九九视频| 日韩成人一级片| 欧美亚洲国产一卡| 亚洲欧洲国产专区| 国产精品一二三区在线| 欧美人妖巨大在线| 亚洲图片一区二区| av综合在线播放| 久久综合久久久久88| 麻豆成人91精品二区三区| 欧美日韩国产a| 亚洲国产成人91porn| 色偷偷久久人人79超碰人人澡| 国产清纯美女被跳蛋高潮一区二区久久w| 天天综合日日夜夜精品| 日本久久一区二区三区| 亚洲免费观看在线视频| 成人免费视频播放| 国产日产欧美一区二区三区| 国内成人免费视频| 精品国产亚洲在线| 久久成人久久鬼色| 2欧美一区二区三区在线观看视频| 秋霞午夜鲁丝一区二区老狼| 91精品视频网| 免费三级欧美电影| 欧美一区二区三区在线| 三级欧美在线一区| 欧美一区国产二区| 奇米色一区二区| 日韩美女视频在线| 国产老妇另类xxxxx| 国产亚洲精品免费| 不卡一区在线观看|