您好,登錄后才能下訂單哦!
小編給大家分享一下Go如何實現(xiàn)HTTP中間人代理的操作示例,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!
golang是一種編譯語言,可以將代碼編譯為機器代碼,編譯后的二進制文件可以直接部署到目標機器而無需額外的依賴,所以golang的性能優(yōu)于其他的解釋性語言,且可以在golang中使用goroutine來實現(xiàn)并發(fā)性,它提供了一個非常優(yōu)雅的goroutine調(diào)度程序系統(tǒng),可以很容易地生成數(shù)百萬個goroutine。
Go HTTP(S)代理庫, 支持中間人代理解密HTTPS
go get github.com/ouqiang/goproxy
package main import ( "net/http" "time" "github.com/ouqiang/goproxy" ) func main() { proxy := goproxy.New() server := &http.Server{ Addr: ":8080", Handler: proxy, ReadTimeout: 1 * time.Minute, WriteTimeout: 1 * time.Minute, } err := server.ListenAndServe() if err != nil { panic(err) } }
curl -x localhost:8080 https://www.baidu.com
系統(tǒng)需導(dǎo)入根證書 mitm-proxy.crt
package main import ( "crypto/tls" "net/http" "sync" "time" "github.com/ouqiang/goproxy" ) // 實現(xiàn)證書緩存接口 type Cache struct { m sync.Map } func (c *Cache) Set(host string, cert *tls.Certificate) { c.m.Store(host, cert) } func (c *Cache) Get(host string) *tls.Certificate { v, ok := c.m.Load(host) if !ok { return nil } return v.(*tls.Certificate) } func main() { proxy := goproxy.New(goproxy.WithDecryptHTTPS(&Cache{})) server := &http.Server{ Addr: ":8080", Handler: proxy, ReadTimeout: 1 * time.Minute, WriteTimeout: 1 * time.Minute, } err := server.ListenAndServe() if err != nil { panic(err) } }
實現(xiàn)Delegate接口
type Delegate interface { // Connect 收到客戶端連接 Connect(ctx *Context, rw http.ResponseWriter) // Auth 代理身份認證 Auth(ctx *Context, rw http.ResponseWriter) // BeforeRequest HTTP請求前 設(shè)置X-Forwarded-For, 修改Header、Body BeforeRequest(ctx *Context) // BeforeResponse 響應(yīng)發(fā)送到客戶端前, 修改Header、Body、Status Code BeforeResponse(ctx *Context, resp *http.Response, err error) // ParentProxy 上級代理 ParentProxy(*http.Request) (*url.URL, error) // Finish 本次請求結(jié)束 Finish(ctx *Context) // 記錄錯誤信息 ErrorLog(err error) }
type EventHandler struct{} func (e *EventHandler) Connect(ctx *goproxy.Context, rw http.ResponseWriter) { // 保存的數(shù)據(jù)可以在后面的回調(diào)方法中獲取 ctx.Data["req_id"] = "uuid" // 禁止訪問某個域名 if strings.Contains(ctx.Req.URL.Host, "example.com") { rw.WriteHeader(http.StatusForbidden) ctx.Abort() return } } func (e *EventHandler) Auth(ctx *goproxy.Context, rw http.ResponseWriter) { // 身份驗證 } func (e *EventHandler) BeforeRequest(ctx *goproxy.Context) { // 修改header ctx.Req.Header.Add("X-Request-Id", ctx.Data["req_id"].(string)) // 設(shè)置X-Forwarded-For if clientIP, _, err := net.SplitHostPort(ctx.Req.RemoteAddr); err == nil { if prior, ok := ctx.Req.Header["X-Forwarded-For"]; ok { clientIP = strings.Join(prior, ", ") + ", " + clientIP } ctx.Req.Header.Set("X-Forwarded-For", clientIP) } // 讀取Body body, err := ioutil.ReadAll(ctx.Req.Body) if err != nil { // 錯誤處理 return } // Request.Body只能讀取一次, 讀取后必須再放回去 // Response.Body同理 ctx.Req.Body = ioutil.NopCloser(bytes.NewReader(body)) } func (e *EventHandler) BeforeResponse(ctx *goproxy.Context, resp *http.Response, err error) { if err != nil { return } // 修改response } // 設(shè)置上級代理 func (e *EventHandler) ParentProxy(req *http.Request) (*url.URL, error) { return url.Parse("http://localhost:1087") } func (e *EventHandler) Finish(ctx *goproxy.Context) { fmt.Printf("請求結(jié)束 URL:%s\n", ctx.Req.URL) } // 記錄錯誤日志 func (e *EventHandler) ErrorLog(err error) { log.Println(err) } func main() { proxy := goproxy.New(goproxy.WithDelegate(&EventHandler{})) server := &http.Server{ Addr: ":8080", Handler: proxy, ReadTimeout: 1 * time.Minute, WriteTimeout: 1 * time.Minute, } err := server.ListenAndServe() if err != nil { panic(err) } }
以上是“Go如何實現(xiàn)HTTP中間人代理的操作示例”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。