Golang:如何阅读ReverseProxy的响应体?

时间:2015-07-21 09:43:40

标签: go reverse-proxy

package main

import (
    "net/http"
    "net/http/httputil"
    "net/url"
)

func main() {
    target := &url.URL{Scheme: "http", Host: "www.google.com"}
    proxy := httputil.NewSingleHostReverseProxy(target)

    http.Handle("/google", proxy)
    http.ListenAndServe(":8099", nil)
}

反向代理是有效的。我怎样才能获得回复正文?

3 个答案:

答案 0 :(得分:20)

httputil.ReverseProxy有一个Transport字段。您可以使用它来修改响应。例如:

type transport struct {
    http.RoundTripper
}

func (t *transport) RoundTrip(req *http.Request) (resp *http.Response, err error) {
    resp, err = t.RoundTripper.RoundTrip(req)
    if err != nil {
        return nil, err
    }
    b, err := ioutil.ReadAll(resp.Body)
    if err != nil {
        return nil, err
    }
    err = resp.Body.Close()
    if err != nil {
        return nil, err
    }
    b = bytes.Replace(b, []byte("server"), []byte("schmerver"), -1)
    body := ioutil.NopCloser(bytes.NewReader(b))
    resp.Body = body
    resp.ContentLength = int64(len(b))
    resp.Header.Set("Content-Length", strconv.Itoa(len(b)))
    return resp, nil
}

// ...
proxy := httputil.NewSingleHostReverseProxy(target)
proxy.Transport = &transport{http.DefaultTransport}

整个事物的游乐场示例:http://play.golang.org/p/b0S5CbCMrI

答案 1 :(得分:4)

我不知道最佳解决方案。但你可以这样做:

package main

import (
    "fmt"
    "net/http"
    "net/http/httputil"
    "net/url"
)

func main() {
    target := &url.URL{Scheme: "http", Host: "www.google.com"}
    proxy := httputil.NewSingleHostReverseProxy(target)

    http.Handle("/google", CustomHandler(proxy))
    http.ListenAndServe(":8099", nil)
}

func CustomHandler(h http.Handler) http.HandlerFunc {
    return func(res http.ResponseWriter, req *http.Request) {
        h.ServeHTTP(NewCustomWriter(res), req)
    }
}

type customWriter struct {
    http.ResponseWriter
}

func NewCustomWriter(w http.ResponseWriter) *customWriter {
    return &customWriter{w}
}

func (c *customWriter) Header() http.Header {
    return c.ResponseWriter.Header()
}

func (c *customWriter) Write(data []byte) (int, error) {
    fmt.Println(string(data)) //get response here
    return c.ResponseWriter.Write(data)
}

func (c *customWriter) WriteHeader(i int) {
    c.ResponseWriter.WriteHeader(i)
}

答案 2 :(得分:0)

从源代码 httptest.ResponseRecorder 用于从处理程序获取响应

func TestModifyResponseClosesBody(t *testing.T) {
    req, _ := http.NewRequest("GET", "http://foo.tld/", nil)
    req.RemoteAddr = "1.2.3.4:56789"
    closeCheck := new(checkCloser)
    logBuf := new(bytes.Buffer)
    outErr := errors.New("ModifyResponse error")
    rp := &ReverseProxy{
        Director: func(req *http.Request) {},
        Transport: &staticTransport{&http.Response{
            StatusCode: 200,
            Body:       closeCheck,
        }},
        ErrorLog: log.New(logBuf, "", 0),
        ModifyResponse: func(*http.Response) error {
            return outErr
        },
    }
    rec := httptest.NewRecorder()
    rp.ServeHTTP(rec, req)
    res := rec.Result()
    if g, e := res.StatusCode, http.StatusBadGateway; g != e {
        t.Errorf("got res.StatusCode %d; expected %d", g, e)
    }
    if !closeCheck.closed {
        t.Errorf("body should have been closed")
    }
    if g, e := logBuf.String(), outErr.Error(); !strings.Contains(g, e) {
        t.Errorf("ErrorLog %q does not contain %q", g, e)
    }
}