問題及場景
業(yè)務(wù)當(dāng)中有需要分發(fā)http.request.body的場景鬓梅。比如微信回調(diào)消息只能指定一個(gè)地址整袁,所以期望可以復(fù)制一份消息發(fā)給其他服務(wù)藤巢。由服務(wù)B和接收微信回調(diào)的服務(wù)A一起處理微信回調(diào)信息鼠次。
解決思路
最開始考慮的是直接轉(zhuǎn)發(fā)http.request慨代。使用ReverseProxy直接將http.request由服務(wù)A轉(zhuǎn)發(fā)給服務(wù)B邢笙。但是微信涉及到驗(yàn)證等問題,完全調(diào)整好非常麻煩侍匙。所以轉(zhuǎn)換思路氮惯,打算將http.request.body的內(nèi)容直接post給服務(wù)B。
可是http.request是readcloser想暗。我們將http.request readAll的時(shí)候講無法再次讀取http.request里面的信息妇汗。
如何才能將http.request.body復(fù)制使用呢?
其中c表示的是http的上下文
// 把request的內(nèi)容讀取出來
var bodyBytes []byte
if c.Request.Body != nil {
bodyBytes, _ = ioutil.ReadAll(c.Request.Body)
}
// 把剛剛讀出來的再寫進(jìn)去
c.Request.Body = ioutil.NopCloser(bytes.NewBuffer(bodyBytes))
1.我們先將body從http.request里面讀取出來说莫,保存到一個(gè)變量里面杨箭。
2.然后再將變量里面的數(shù)據(jù)使用ioutil.NopCloser方法寫回到http.request里面。
https://golang.org/pkg/io/ioutil/#NopCloser
NopCloser returns a ReadCloser with a no-op Close method wrapping the provided Reader r.
NopCloser用一個(gè)無操作的Close方法包裝Reader r返回一個(gè)ReadCloser接口储狭。
這樣我們就可以再次使用c.request來進(jìn)行處理了互婿。