mirror of
https://github.com/labstack/echo.git
synced 2025-07-03 00:56:59 +02:00
This commit is contained in:
@ -2,7 +2,6 @@ package middleware
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/url"
|
|
||||||
"regexp"
|
"regexp"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
@ -49,30 +48,39 @@ func rewriteRulesRegex(rewrite map[string]string) map[*regexp.Regexp]string {
|
|||||||
return rulesRegex
|
return rulesRegex
|
||||||
}
|
}
|
||||||
|
|
||||||
func rewritePath(rewriteRegex map[*regexp.Regexp]string, req *http.Request) {
|
func rewriteURL(rewriteRegex map[*regexp.Regexp]string, req *http.Request) error {
|
||||||
|
if len(rewriteRegex) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Depending how HTTP request is sent RequestURI could contain Scheme://Host/path or be just /path.
|
||||||
|
// We only want to use path part for rewriting and therefore trim prefix if it exists
|
||||||
|
rawURI := req.RequestURI
|
||||||
|
if rawURI != "" && rawURI[0] != '/' {
|
||||||
|
prefix := ""
|
||||||
|
if req.URL.Scheme != "" {
|
||||||
|
prefix = req.URL.Scheme + "://"
|
||||||
|
}
|
||||||
|
if req.URL.Host != "" {
|
||||||
|
prefix += req.URL.Host // host or host:port
|
||||||
|
}
|
||||||
|
if prefix != "" {
|
||||||
|
rawURI = strings.TrimPrefix(rawURI, prefix)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
for k, v := range rewriteRegex {
|
for k, v := range rewriteRegex {
|
||||||
rawPath := req.URL.RawPath
|
if replacer := captureTokens(k, rawURI); replacer != nil {
|
||||||
if rawPath != "" {
|
url, err := req.URL.Parse(replacer.Replace(v))
|
||||||
// RawPath is only set when there has been escaping done. In that case Path must be deduced from rewritten RawPath
|
if err != nil {
|
||||||
// because encoded Path could match rules that RawPath did not
|
return err
|
||||||
if replacer := captureTokens(k, rawPath); replacer != nil {
|
|
||||||
rawPath = replacer.Replace(v)
|
|
||||||
|
|
||||||
req.URL.RawPath = rawPath
|
|
||||||
req.URL.Path, _ = url.PathUnescape(rawPath)
|
|
||||||
|
|
||||||
return // rewrite only once
|
|
||||||
}
|
}
|
||||||
|
req.URL = url
|
||||||
|
|
||||||
continue
|
return nil // rewrite only once
|
||||||
}
|
|
||||||
|
|
||||||
if replacer := captureTokens(k, req.URL.Path); replacer != nil {
|
|
||||||
req.URL.Path = replacer.Replace(v)
|
|
||||||
|
|
||||||
return // rewrite only once
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// DefaultSkipper returns false which processes the middleware.
|
// DefaultSkipper returns false which processes the middleware.
|
||||||
|
@ -8,11 +8,13 @@ import (
|
|||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestRewritePath(t *testing.T) {
|
func TestRewriteURL(t *testing.T) {
|
||||||
var testCases = []struct {
|
var testCases = []struct {
|
||||||
whenURL string
|
whenURL string
|
||||||
expectPath string
|
expectPath string
|
||||||
expectRawPath string
|
expectRawPath string
|
||||||
|
expectQuery string
|
||||||
|
expectErr string
|
||||||
}{
|
}{
|
||||||
{
|
{
|
||||||
whenURL: "http://localhost:8080/old",
|
whenURL: "http://localhost:8080/old",
|
||||||
@ -28,6 +30,7 @@ func TestRewritePath(t *testing.T) {
|
|||||||
whenURL: "http://localhost:8080/users/+_+/orders/___++++?test=1",
|
whenURL: "http://localhost:8080/users/+_+/orders/___++++?test=1",
|
||||||
expectPath: "/user/+_+/order/___++++",
|
expectPath: "/user/+_+/order/___++++",
|
||||||
expectRawPath: "",
|
expectRawPath: "",
|
||||||
|
expectQuery: "test=1",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
whenURL: "http://localhost:8080/users/%20a/orders/%20aa",
|
whenURL: "http://localhost:8080/users/%20a/orders/%20aa",
|
||||||
@ -35,9 +38,10 @@ func TestRewritePath(t *testing.T) {
|
|||||||
expectRawPath: "",
|
expectRawPath: "",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
whenURL: "http://localhost:8080/%47%6f%2f",
|
whenURL: "http://localhost:8080/%47%6f%2f?test=1",
|
||||||
expectPath: "/Go/",
|
expectPath: "/Go/",
|
||||||
expectRawPath: "/%47%6f%2f",
|
expectRawPath: "/%47%6f%2f",
|
||||||
|
expectQuery: "test=1",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
whenURL: "/users/jill/orders/T%2FcO4lW%2Ft%2FVp%2F",
|
whenURL: "/users/jill/orders/T%2FcO4lW%2Ft%2FVp%2F",
|
||||||
@ -49,21 +53,40 @@ func TestRewritePath(t *testing.T) {
|
|||||||
expectPath: "/user/jill/order/T/cO4lW/t/Vp/",
|
expectPath: "/user/jill/order/T/cO4lW/t/Vp/",
|
||||||
expectRawPath: "/user/jill/order/T%2FcO4lW%2Ft%2FVp%2F",
|
expectRawPath: "/user/jill/order/T%2FcO4lW%2Ft%2FVp%2F",
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
whenURL: "http://localhost:8080/static",
|
||||||
|
expectPath: "/static/path",
|
||||||
|
expectRawPath: "",
|
||||||
|
expectQuery: "role=AUTHOR&limit=1000",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
whenURL: "/static",
|
||||||
|
expectPath: "/static/path",
|
||||||
|
expectRawPath: "",
|
||||||
|
expectQuery: "role=AUTHOR&limit=1000",
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
rules := map[*regexp.Regexp]string{
|
rules := map[*regexp.Regexp]string{
|
||||||
regexp.MustCompile("^/old$"): "/new",
|
regexp.MustCompile("^/old$"): "/new",
|
||||||
regexp.MustCompile("^/users/(.*?)/orders/(.*?)$"): "/user/$1/order/$2",
|
regexp.MustCompile("^/users/(.*?)/orders/(.*?)$"): "/user/$1/order/$2",
|
||||||
|
regexp.MustCompile("^/static$"): "/static/path?role=AUTHOR&limit=1000",
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.whenURL, func(t *testing.T) {
|
t.Run(tc.whenURL, func(t *testing.T) {
|
||||||
req := httptest.NewRequest(http.MethodGet, tc.whenURL, nil)
|
req := httptest.NewRequest(http.MethodGet, tc.whenURL, nil)
|
||||||
|
|
||||||
rewritePath(rules, req)
|
err := rewriteURL(rules, req)
|
||||||
|
|
||||||
|
if tc.expectErr != "" {
|
||||||
|
assert.EqualError(t, err, tc.expectErr)
|
||||||
|
} else {
|
||||||
|
assert.NoError(t, err)
|
||||||
|
}
|
||||||
assert.Equal(t, tc.expectPath, req.URL.Path) // Path field is stored in decoded form: /%47%6f%2f becomes /Go/.
|
assert.Equal(t, tc.expectPath, req.URL.Path) // Path field is stored in decoded form: /%47%6f%2f becomes /Go/.
|
||||||
assert.Equal(t, tc.expectRawPath, req.URL.RawPath) // RawPath, an optional field which only gets set if the default encoding is different from Path.
|
assert.Equal(t, tc.expectRawPath, req.URL.RawPath) // RawPath, an optional field which only gets set if the default encoding is different from Path.
|
||||||
|
assert.Equal(t, tc.expectQuery, req.URL.RawQuery)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -231,8 +231,9 @@ func ProxyWithConfig(config ProxyConfig) echo.MiddlewareFunc {
|
|||||||
tgt := config.Balancer.Next(c)
|
tgt := config.Balancer.Next(c)
|
||||||
c.Set(config.ContextKey, tgt)
|
c.Set(config.ContextKey, tgt)
|
||||||
|
|
||||||
// Set rewrite path and raw path
|
if err := rewriteURL(config.RegexRewrite, req); err != nil {
|
||||||
rewritePath(config.RegexRewrite, req)
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
// Fix header
|
// Fix header
|
||||||
// Basically it's not good practice to unconditionally pass incoming x-real-ip header to upstream.
|
// Basically it's not good practice to unconditionally pass incoming x-real-ip header to upstream.
|
||||||
|
@ -245,12 +245,16 @@ func TestProxyRewrite(t *testing.T) {
|
|||||||
|
|
||||||
func TestProxyRewriteRegex(t *testing.T) {
|
func TestProxyRewriteRegex(t *testing.T) {
|
||||||
// Setup
|
// Setup
|
||||||
upstream := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {}))
|
receivedRequestURI := make(chan string, 1)
|
||||||
|
upstream := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
// RequestURI is the unmodified request-target of the Request-Line (RFC 7230, Section 3.1.1) as sent by the client to a server
|
||||||
|
// we need unmodified target to see if we are encoding/decoding the url in addition to rewrite/replace logic
|
||||||
|
// if original request had `%2F` we should not magically decode it to `/` as it would change what was requested
|
||||||
|
receivedRequestURI <- r.RequestURI
|
||||||
|
}))
|
||||||
defer upstream.Close()
|
defer upstream.Close()
|
||||||
url, _ := url.Parse(upstream.URL)
|
tmpUrL, _ := url.Parse(upstream.URL)
|
||||||
rrb := NewRoundRobinBalancer([]*ProxyTarget{{Name: "upstream", URL: url}})
|
rrb := NewRoundRobinBalancer([]*ProxyTarget{{Name: "upstream", URL: tmpUrL}})
|
||||||
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
||||||
rec := httptest.NewRecorder()
|
|
||||||
|
|
||||||
// Rewrite
|
// Rewrite
|
||||||
e := echo.New()
|
e := echo.New()
|
||||||
@ -279,14 +283,21 @@ func TestProxyRewriteRegex(t *testing.T) {
|
|||||||
{"/c/ignore1/test/this", http.StatusOK, "/v3/test/this"},
|
{"/c/ignore1/test/this", http.StatusOK, "/v3/test/this"},
|
||||||
{"/x/ignore/test", http.StatusOK, "/v4/test"},
|
{"/x/ignore/test", http.StatusOK, "/v4/test"},
|
||||||
{"/y/foo/bar", http.StatusOK, "/v5/bar/foo"},
|
{"/y/foo/bar", http.StatusOK, "/v5/bar/foo"},
|
||||||
|
// NB: fragment is not added by golang httputil.NewSingleHostReverseProxy implementation
|
||||||
|
// $2 = `bar?q=1#frag`, $1 = `foo`. replaced uri = `/v5/bar?q=1#frag/foo` but httputil.NewSingleHostReverseProxy does not send `#frag/foo` (currently)
|
||||||
|
{"/y/foo/bar?q=1#frag", http.StatusOK, "/v5/bar?q=1"},
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
t.Run(tc.requestPath, func(t *testing.T) {
|
t.Run(tc.requestPath, func(t *testing.T) {
|
||||||
req.URL, _ = url.Parse(tc.requestPath)
|
targetURL, _ := url.Parse(tc.requestPath)
|
||||||
rec = httptest.NewRecorder()
|
req := httptest.NewRequest(http.MethodGet, targetURL.String(), nil)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
|
||||||
e.ServeHTTP(rec, req)
|
e.ServeHTTP(rec, req)
|
||||||
assert.Equal(t, tc.expectPath, req.URL.EscapedPath())
|
|
||||||
|
actualRequestURI := <-receivedRequestURI
|
||||||
|
assert.Equal(t, tc.expectPath, actualRequestURI)
|
||||||
assert.Equal(t, tc.statusCode, rec.Code)
|
assert.Equal(t, tc.statusCode, rec.Code)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -72,9 +72,9 @@ func RewriteWithConfig(config RewriteConfig) echo.MiddlewareFunc {
|
|||||||
return next(c)
|
return next(c)
|
||||||
}
|
}
|
||||||
|
|
||||||
req := c.Request()
|
if err := rewriteURL(config.RegexRules, c.Request()); err != nil {
|
||||||
// Set rewrite path and raw path
|
return err
|
||||||
rewritePath(config.RegexRules, req)
|
}
|
||||||
return next(c)
|
return next(c)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -220,6 +220,8 @@ func TestEchoRewriteWithRegexRules(t *testing.T) {
|
|||||||
func TestEchoRewriteReplacementEscaping(t *testing.T) {
|
func TestEchoRewriteReplacementEscaping(t *testing.T) {
|
||||||
e := echo.New()
|
e := echo.New()
|
||||||
|
|
||||||
|
// NOTE: these are incorrect regexps as they do not factor in that URI we are replacing could contain ? (query) and # (fragment) parts
|
||||||
|
// so in reality they append query and fragment part as `$1` matches everything after that prefix
|
||||||
e.Pre(RewriteWithConfig(RewriteConfig{
|
e.Pre(RewriteWithConfig(RewriteConfig{
|
||||||
Rules: map[string]string{
|
Rules: map[string]string{
|
||||||
"^/a/*": "/$1?query=param",
|
"^/a/*": "/$1?query=param",
|
||||||
@ -228,6 +230,7 @@ func TestEchoRewriteReplacementEscaping(t *testing.T) {
|
|||||||
RegexRules: map[*regexp.Regexp]string{
|
RegexRules: map[*regexp.Regexp]string{
|
||||||
regexp.MustCompile("^/x/(.*)"): "/$1?query=param",
|
regexp.MustCompile("^/x/(.*)"): "/$1?query=param",
|
||||||
regexp.MustCompile("^/y/(.*)"): "/$1;part#one",
|
regexp.MustCompile("^/y/(.*)"): "/$1;part#one",
|
||||||
|
regexp.MustCompile("^/z/(.*)"): "/$1?test=1#escaped%20test",
|
||||||
},
|
},
|
||||||
}))
|
}))
|
||||||
|
|
||||||
@ -236,13 +239,15 @@ func TestEchoRewriteReplacementEscaping(t *testing.T) {
|
|||||||
|
|
||||||
testCases := []struct {
|
testCases := []struct {
|
||||||
requestPath string
|
requestPath string
|
||||||
expectPath string
|
expect string
|
||||||
}{
|
}{
|
||||||
{"/unmatched", "/unmatched"},
|
{"/unmatched", "/unmatched"},
|
||||||
{"/a/test", "/test?query=param"},
|
{"/a/test", "/test?query=param"},
|
||||||
{"/b/foo/bar", "/foo/bar;part#one"},
|
{"/b/foo/bar", "/foo/bar;part#one"},
|
||||||
{"/x/test", "/test?query=param"},
|
{"/x/test", "/test?query=param"},
|
||||||
{"/y/foo/bar", "/foo/bar;part#one"},
|
{"/y/foo/bar", "/foo/bar;part#one"},
|
||||||
|
{"/z/foo/b%20ar", "/foo/b%20ar?test=1#escaped%20test"},
|
||||||
|
{"/z/foo/b%20ar?nope=1#yes", "/foo/b%20ar?nope=1#yes?test=1%23escaped%20test"}, // example of appending
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, tc := range testCases {
|
for _, tc := range testCases {
|
||||||
@ -250,7 +255,7 @@ func TestEchoRewriteReplacementEscaping(t *testing.T) {
|
|||||||
req = httptest.NewRequest(http.MethodGet, tc.requestPath, nil)
|
req = httptest.NewRequest(http.MethodGet, tc.requestPath, nil)
|
||||||
rec = httptest.NewRecorder()
|
rec = httptest.NewRecorder()
|
||||||
e.ServeHTTP(rec, req)
|
e.ServeHTTP(rec, req)
|
||||||
assert.Equal(t, tc.expectPath, req.URL.Path)
|
assert.Equal(t, tc.expect, req.URL.String())
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user