2017-12-28 11:24:34 -08:00
package middleware
import (
2018-08-15 16:35:27 +02:00
"io/ioutil"
2018-10-14 17:16:58 +02:00
"net/http"
2017-12-28 11:24:34 -08:00
"net/http/httptest"
2020-09-03 00:39:57 -07:00
"net/url"
2021-02-08 15:58:55 +01:00
"regexp"
2017-12-28 11:24:34 -08:00
"testing"
2021-07-15 23:34:01 +03:00
"github.com/labstack/echo/v5"
2017-12-28 11:24:34 -08:00
"github.com/stretchr/testify/assert"
)
2021-03-08 03:01:02 +02:00
func TestRewriteAfterRouting ( t * testing . T ) {
2017-12-28 11:24:34 -08:00
e := echo . New ( )
2021-03-08 03:01:02 +02:00
// middlewares added with `Use()` are executed after routing is done and do not affect which route handler is matched
2017-12-28 11:24:34 -08:00
e . Use ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string {
"/old" : "/new" ,
"/api/*" : "/$1" ,
"/js/*" : "/public/javascripts/$1" ,
"/users/*/orders/*" : "/user/$1/order/$2" ,
} ,
} ) )
2021-03-08 03:01:02 +02:00
e . GET ( "/public/*" , func ( c echo . Context ) error {
2021-07-15 23:34:01 +03:00
return c . String ( http . StatusOK , c . PathParam ( "*" ) )
2021-03-08 03:01:02 +02:00
} )
e . GET ( "/*" , func ( c echo . Context ) error {
2021-07-15 23:34:01 +03:00
return c . String ( http . StatusOK , c . PathParam ( "*" ) )
2021-03-08 03:01:02 +02:00
} )
var testCases = [ ] struct {
whenPath string
expectRoutePath string
expectRequestPath string
expectRequestRawPath string
} {
{
whenPath : "/api/users" ,
expectRoutePath : "api/users" ,
expectRequestPath : "/users" ,
expectRequestRawPath : "" ,
} ,
{
whenPath : "/js/main.js" ,
expectRoutePath : "js/main.js" ,
expectRequestPath : "/public/javascripts/main.js" ,
expectRequestRawPath : "" ,
} ,
{
whenPath : "/users/jack/orders/1" ,
expectRoutePath : "users/jack/orders/1" ,
expectRequestPath : "/user/jack/order/1" ,
expectRequestRawPath : "" ,
} ,
{ // no rewrite rule matched. already encoded URL should not be double encoded or changed in any way
whenPath : "/user/jill/order/T%2FcO4lW%2Ft%2FVp%2F" ,
expectRoutePath : "user/jill/order/T%2FcO4lW%2Ft%2FVp%2F" ,
expectRequestPath : "/user/jill/order/T/cO4lW/t/Vp/" , // this is equal to `url.Parse(tc.whenPath)` result
expectRequestRawPath : "/user/jill/order/T%2FcO4lW%2Ft%2FVp%2F" ,
} ,
{ // just rewrite but do not touch encoding. already encoded URL should not be double encoded
whenPath : "/users/jill/orders/T%2FcO4lW%2Ft%2FVp%2F" ,
expectRoutePath : "users/jill/orders/T%2FcO4lW%2Ft%2FVp%2F" ,
expectRequestPath : "/user/jill/order/T/cO4lW/t/Vp/" , // this is equal to `url.Parse(tc.whenPath)` result
expectRequestRawPath : "/user/jill/order/T%2FcO4lW%2Ft%2FVp%2F" ,
} ,
{ // ` ` (space) is encoded by httpClient to `%20` when doing request to Echo. `%20` should not be double escaped or changed in any way when rewriting request
whenPath : "/api/new users" ,
expectRoutePath : "api/new users" ,
expectRequestPath : "/new users" ,
expectRequestRawPath : "" ,
} ,
}
for _ , tc := range testCases {
t . Run ( tc . whenPath , func ( t * testing . T ) {
target , _ := url . Parse ( tc . whenPath )
req := httptest . NewRequest ( http . MethodGet , target . String ( ) , nil )
rec := httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
assert . Equal ( t , http . StatusOK , rec . Code )
assert . Equal ( t , tc . expectRoutePath , rec . Body . String ( ) )
assert . Equal ( t , tc . expectRequestPath , req . URL . Path )
assert . Equal ( t , tc . expectRequestRawPath , req . URL . RawPath )
} )
}
2017-12-28 11:24:34 -08:00
}
2018-03-16 15:05:22 +01:00
2021-07-15 23:34:01 +03:00
func TestMustRewriteWithConfig_emptyRulesPanics ( t * testing . T ) {
assert . Panics ( t , func ( ) {
RewriteWithConfig ( RewriteConfig { } )
} )
}
func TestMustRewriteWithConfig_skipper ( t * testing . T ) {
var testCases = [ ] struct {
name string
givenSkipper func ( c echo . Context ) bool
whenURL string
expectURL string
expectStatus int
} {
{
name : "not skipped" ,
whenURL : "/old" ,
expectURL : "/new" ,
expectStatus : http . StatusOK ,
} ,
{
name : "skipped" ,
givenSkipper : func ( c echo . Context ) bool {
return true
} ,
whenURL : "/old" ,
expectURL : "/old" ,
expectStatus : http . StatusNotFound ,
} ,
}
for _ , tc := range testCases {
t . Run ( tc . name , func ( t * testing . T ) {
e := echo . New ( )
e . Pre ( RewriteWithConfig (
RewriteConfig {
Skipper : tc . givenSkipper ,
Rules : map [ string ] string { "/old" : "/new" } } ,
) )
e . GET ( "/new" , func ( c echo . Context ) error {
return c . NoContent ( http . StatusOK )
} )
req := httptest . NewRequest ( http . MethodGet , tc . whenURL , nil )
rec := httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
assert . Equal ( t , tc . expectURL , req . URL . EscapedPath ( ) )
assert . Equal ( t , tc . expectStatus , rec . Code )
} )
}
}
2018-03-16 10:00:59 -07:00
// Issue #1086
2018-03-16 15:05:22 +01:00
func TestEchoRewritePreMiddleware ( t * testing . T ) {
e := echo . New ( )
// Rewrite old url to new one
2021-03-08 03:01:02 +02:00
// middlewares added with `Pre()` are executed before routing is done and therefore change which handler matches
2021-07-15 23:34:01 +03:00
e . Pre ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string { "/old" : "/new" } } ) ,
)
2018-03-16 15:05:22 +01:00
// Route
2021-07-15 23:34:01 +03:00
e . Add ( http . MethodGet , "/new" , func ( c echo . Context ) error {
2020-08-23 11:18:31 +08:00
return c . NoContent ( http . StatusOK )
2018-03-16 15:05:22 +01:00
} )
2018-10-14 17:16:58 +02:00
req := httptest . NewRequest ( http . MethodGet , "/old" , nil )
2018-03-16 15:05:22 +01:00
rec := httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
2020-08-28 12:47:02 -07:00
assert . Equal ( t , "/new" , req . URL . EscapedPath ( ) )
2020-08-23 11:18:31 +08:00
assert . Equal ( t , http . StatusOK , rec . Code )
2018-03-16 15:05:22 +01:00
}
2018-08-15 16:35:27 +02:00
// Issue #1143
func TestRewriteWithConfigPreMiddleware_Issue1143 ( t * testing . T ) {
e := echo . New ( )
2021-03-08 03:01:02 +02:00
// middlewares added with `Pre()` are executed before routing is done and therefore change which handler matches
2018-08-15 16:35:27 +02:00
e . Pre ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string {
"/api/*/mgmt/proj/*/agt" : "/api/$1/hosts/$2" ,
"/api/*/mgmt/proj" : "/api/$1/eng" ,
} ,
} ) )
2021-07-15 23:34:01 +03:00
e . Add ( http . MethodGet , "/api/:version/hosts/:name" , func ( c echo . Context ) error {
2020-08-23 11:18:31 +08:00
return c . String ( http . StatusOK , "hosts" )
2018-08-15 16:35:27 +02:00
} )
2021-07-15 23:34:01 +03:00
e . Add ( http . MethodGet , "/api/:version/eng" , func ( c echo . Context ) error {
2020-08-23 11:18:31 +08:00
return c . String ( http . StatusOK , "eng" )
2018-08-15 16:35:27 +02:00
} )
for i := 0 ; i < 100 ; i ++ {
2018-10-14 17:16:58 +02:00
req := httptest . NewRequest ( http . MethodGet , "/api/v1/mgmt/proj/test/agt" , nil )
2018-08-15 16:35:27 +02:00
rec := httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
2020-08-28 12:47:02 -07:00
assert . Equal ( t , "/api/v1/hosts/test" , req . URL . EscapedPath ( ) )
2020-08-23 11:18:31 +08:00
assert . Equal ( t , http . StatusOK , rec . Code )
2018-08-15 16:35:27 +02:00
defer rec . Result ( ) . Body . Close ( )
bodyBytes , _ := ioutil . ReadAll ( rec . Result ( ) . Body )
assert . Equal ( t , "hosts" , string ( bodyBytes ) )
}
}
2020-07-24 22:01:19 +08:00
// Issue #1573
func TestEchoRewriteWithCaret ( t * testing . T ) {
e := echo . New ( )
e . Pre ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string {
"^/abc/*" : "/v1/abc/$1" ,
} ,
} ) )
rec := httptest . NewRecorder ( )
var req * http . Request
req = httptest . NewRequest ( http . MethodGet , "/abc/test" , nil )
e . ServeHTTP ( rec , req )
assert . Equal ( t , "/v1/abc/test" , req . URL . Path )
req = httptest . NewRequest ( http . MethodGet , "/v1/abc/test" , nil )
e . ServeHTTP ( rec , req )
assert . Equal ( t , "/v1/abc/test" , req . URL . Path )
req = httptest . NewRequest ( http . MethodGet , "/v2/abc/test" , nil )
e . ServeHTTP ( rec , req )
assert . Equal ( t , "/v2/abc/test" , req . URL . Path )
}
2021-02-08 15:58:55 +01:00
// Verify regex used with rewrite
func TestEchoRewriteWithRegexRules ( t * testing . T ) {
e := echo . New ( )
e . Pre ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string {
"^/a/*" : "/v1/$1" ,
"^/b/*/c/*" : "/v2/$2/$1" ,
"^/c/*/*" : "/v3/$2" ,
} ,
RegexRules : map [ * regexp . Regexp ] string {
regexp . MustCompile ( "^/x/.+?/(.*)" ) : "/v4/$1" ,
regexp . MustCompile ( "^/y/(.+?)/(.*)" ) : "/v5/$2/$1" ,
} ,
} ) )
var rec * httptest . ResponseRecorder
var req * http . Request
testCases := [ ] struct {
requestPath string
expectPath string
} {
{ "/unmatched" , "/unmatched" } ,
{ "/a/test" , "/v1/test" } ,
{ "/b/foo/c/bar/baz" , "/v2/bar/baz/foo" } ,
{ "/c/ignore/test" , "/v3/test" } ,
{ "/c/ignore1/test/this" , "/v3/test/this" } ,
{ "/x/ignore/test" , "/v4/test" } ,
{ "/y/foo/bar" , "/v5/bar/foo" } ,
}
2021-02-27 06:55:00 +09:00
for _ , tc := range testCases {
t . Run ( tc . requestPath , func ( t * testing . T ) {
req = httptest . NewRequest ( http . MethodGet , tc . requestPath , nil )
rec = httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
assert . Equal ( t , tc . expectPath , req . URL . EscapedPath ( ) )
} )
}
2021-02-08 15:58:55 +01:00
}
2021-03-08 03:01:02 +02:00
// Ensure correct escaping as defined in replacement (issue #1798)
func TestEchoRewriteReplacementEscaping ( t * testing . T ) {
e := echo . New ( )
2021-03-09 14:22:11 +02:00
// 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
2021-03-08 03:01:02 +02:00
e . Pre ( RewriteWithConfig ( RewriteConfig {
Rules : map [ string ] string {
"^/a/*" : "/$1?query=param" ,
"^/b/*" : "/$1;part#one" ,
} ,
RegexRules : map [ * regexp . Regexp ] string {
regexp . MustCompile ( "^/x/(.*)" ) : "/$1?query=param" ,
regexp . MustCompile ( "^/y/(.*)" ) : "/$1;part#one" ,
2021-03-09 14:22:11 +02:00
regexp . MustCompile ( "^/z/(.*)" ) : "/$1?test=1#escaped%20test" ,
2021-03-08 03:01:02 +02:00
} ,
} ) )
var rec * httptest . ResponseRecorder
var req * http . Request
testCases := [ ] struct {
requestPath string
2021-03-09 14:22:11 +02:00
expect string
2021-03-08 03:01:02 +02:00
} {
{ "/unmatched" , "/unmatched" } ,
{ "/a/test" , "/test?query=param" } ,
{ "/b/foo/bar" , "/foo/bar;part#one" } ,
{ "/x/test" , "/test?query=param" } ,
{ "/y/foo/bar" , "/foo/bar;part#one" } ,
2021-03-09 14:22:11 +02:00
{ "/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
2021-03-08 03:01:02 +02:00
}
for _ , tc := range testCases {
t . Run ( tc . requestPath , func ( t * testing . T ) {
req = httptest . NewRequest ( http . MethodGet , tc . requestPath , nil )
rec = httptest . NewRecorder ( )
e . ServeHTTP ( rec , req )
2021-03-09 14:22:11 +02:00
assert . Equal ( t , tc . expect , req . URL . String ( ) )
2021-03-08 03:01:02 +02:00
} )
}
}