mirror of
https://github.com/labstack/echo.git
synced 2025-01-07 23:01:56 +02:00
6ef5f77bf2
WIP: make default logger implemented custom writer for jsonlike logs WIP: improve examples WIP: defaultErrorHandler use errors.As to unwrap errors. Update readme WIP: default logger logs json, restore e.Start method WIP: clean router.Match a bit WIP: func types/fields have echo.Context has first element WIP: remove yaml tags as functions etc can not be serialized anyway WIP: change BindPathParams,BindQueryParams,BindHeaders from methods to functions and reverse arguments to be like DefaultBinder.Bind is WIP: improved comments, logger now extracts status from error WIP: go mod tidy WIP: rebase with 4.5.0 WIP: * removed todos. * removed StartAutoTLS and StartH2CServer methods from `StartConfig` * KeyAuth middleware errorhandler can swallow the error and resume next middleware WIP: add RouterConfig.UseEscapedPathForMatching to use escaped path for matching request against routes WIP: FIXMEs WIP: upgrade golang-jwt/jwt to `v4` WIP: refactor http methods to return RouteInfo WIP: refactor static not creating multiple routes WIP: refactor route and middleware adding functions not to return error directly WIP: Use 401 for problematic/missing headers for key auth and JWT middleware (#1552, #1402). > In summary, a 401 Unauthorized response should be used for missing or bad authentication WIP: replace `HTTPError.SetInternal` with `HTTPError.WithInternal` so we could not mutate global error variables WIP: add RouteInfo and RouteMatchType into Context what we could know from in middleware what route was matched and/or type of that match (200/404/405) WIP: make notFoundHandler and methodNotAllowedHandler private. encourage that all errors be handled in Echo.HTTPErrorHandler WIP: server cleanup ideas WIP: routable.ForGroup WIP: note about logger middleware WIP: bind should not default values on second try. use crypto rand for better randomness WIP: router add route as interface and returns info as interface WIP: improve flaky test (remains still flaky) WIP: add notes about bind default values WIP: every route can have their own path params names WIP: routerCreator and different tests WIP: different things WIP: remove route implementation WIP: support custom method types WIP: extractor tests WIP: v5.0.x proposal over v4.4.0
1014 lines
27 KiB
Go
1014 lines
27 KiB
Go
package echo
|
|
|
|
import (
|
|
"bytes"
|
|
stdContext "context"
|
|
"errors"
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
)
|
|
|
|
type user struct {
|
|
ID int `json:"id" xml:"id" form:"id" query:"id" param:"id" header:"id"`
|
|
Name string `json:"name" xml:"name" form:"name" query:"name" param:"name" header:"name"`
|
|
}
|
|
|
|
const (
|
|
userJSON = `{"id":1,"name":"Jon Snow"}`
|
|
usersJSON = `[{"id":1,"name":"Jon Snow"}]`
|
|
userXML = `<user><id>1</id><name>Jon Snow</name></user>`
|
|
userForm = `id=1&name=Jon Snow`
|
|
invalidContent = "invalid content"
|
|
userJSONInvalidType = `{"id":"1","name":"Jon Snow"}`
|
|
userXMLConvertNumberError = `<user><id>Number one</id><name>Jon Snow</name></user>`
|
|
userXMLUnsupportedTypeError = `<user><>Number one</><name>Jon Snow</name></user>`
|
|
)
|
|
|
|
const userJSONPretty = `{
|
|
"id": 1,
|
|
"name": "Jon Snow"
|
|
}`
|
|
|
|
const userXMLPretty = `<user>
|
|
<id>1</id>
|
|
<name>Jon Snow</name>
|
|
</user>`
|
|
|
|
var dummyQuery = url.Values{"dummy": []string{"useless"}}
|
|
|
|
func TestEcho(t *testing.T) {
|
|
e := New()
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
rec := httptest.NewRecorder()
|
|
c := e.NewContext(req, rec)
|
|
|
|
// Router
|
|
assert.NotNil(t, e.Router())
|
|
|
|
e.HTTPErrorHandler(c, errors.New("error"))
|
|
|
|
assert.Equal(t, http.StatusInternalServerError, rec.Code)
|
|
}
|
|
|
|
func TestEchoStatic(t *testing.T) {
|
|
var testCases = []struct {
|
|
name string
|
|
givenPrefix string
|
|
givenRoot string
|
|
whenURL string
|
|
expectStatus int
|
|
expectHeaderLocation string
|
|
expectBodyStartsWith string
|
|
}{
|
|
{
|
|
name: "ok",
|
|
givenPrefix: "/images",
|
|
givenRoot: "_fixture/images",
|
|
whenURL: "/images/walle.png",
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: string([]byte{0x89, 0x50, 0x4e, 0x47}),
|
|
},
|
|
{
|
|
name: "without prefix",
|
|
givenPrefix: "",
|
|
givenRoot: "_fixture/images",
|
|
whenURL: "/walle.png", // `` + `*` creates route `/test*` witch matches `walle.png`
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: string([]byte{0x89, 0x50, 0x4e, 0x47}),
|
|
},
|
|
{
|
|
name: "without prefix does not serve dir index or redirect",
|
|
givenPrefix: "",
|
|
givenRoot: "_fixture/images",
|
|
whenURL: "/", // `/` + `*` creates route `/*`
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
{
|
|
name: "No file",
|
|
givenPrefix: "/images",
|
|
givenRoot: "_fixture/scripts",
|
|
whenURL: "/images/bolt.png",
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
{
|
|
name: "Directory",
|
|
givenPrefix: "/images",
|
|
givenRoot: "_fixture/images",
|
|
whenURL: "/images/",
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
{
|
|
name: "Directory Redirect",
|
|
givenPrefix: "/",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/folder", // `/folder` is subdirectory inside `/_fixture`
|
|
expectStatus: http.StatusMovedPermanently,
|
|
expectHeaderLocation: "/folder/",
|
|
expectBodyStartsWith: "",
|
|
},
|
|
{
|
|
name: "Directory Redirect with non-root path",
|
|
givenPrefix: "/static",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/static",
|
|
expectStatus: http.StatusMovedPermanently,
|
|
expectHeaderLocation: "/static/",
|
|
expectBodyStartsWith: "",
|
|
},
|
|
{
|
|
name: "Prefixed directory 404 (request URL without slash)",
|
|
givenPrefix: "/folder/", // trailing slash will intentionally not match "/folder"
|
|
givenRoot: "_fixture",
|
|
whenURL: "/folder", // no trailing slash
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
{
|
|
name: "Prefixed directory redirect (without slash redirect to slash)",
|
|
givenPrefix: "/folder", // no trailing slash shall match /folder and /folder/*
|
|
givenRoot: "_fixture",
|
|
whenURL: "/folder", // no trailing slash
|
|
expectStatus: http.StatusMovedPermanently,
|
|
expectHeaderLocation: "/folder/",
|
|
expectBodyStartsWith: "",
|
|
},
|
|
{
|
|
name: "Directory with index.html",
|
|
givenPrefix: "/",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/",
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: "<!doctype html>",
|
|
},
|
|
{
|
|
name: "Prefixed directory with index.html (prefix ending with slash)",
|
|
givenPrefix: "/assets/",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/assets/",
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: "<!doctype html>",
|
|
},
|
|
{
|
|
name: "Prefixed directory with index.html (prefix ending without slash)",
|
|
givenPrefix: "/assets",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/assets/",
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: "<!doctype html>",
|
|
},
|
|
{
|
|
name: "Sub-directory with index.html",
|
|
givenPrefix: "/",
|
|
givenRoot: "_fixture",
|
|
whenURL: "/folder/",
|
|
expectStatus: http.StatusOK,
|
|
expectBodyStartsWith: "<!doctype html>",
|
|
},
|
|
{
|
|
name: "do not allow directory traversal (backslash - windows separator)",
|
|
givenPrefix: "/",
|
|
givenRoot: "_fixture/",
|
|
whenURL: `/..\\middleware/basic_auth.go`,
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
{
|
|
name: "do not allow directory traversal (slash - unix separator)",
|
|
givenPrefix: "/",
|
|
givenRoot: "_fixture/",
|
|
whenURL: `/../middleware/basic_auth.go`,
|
|
expectStatus: http.StatusNotFound,
|
|
expectBodyStartsWith: "{\"message\":\"Not Found\"}\n",
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
e := New()
|
|
e.Static(tc.givenPrefix, tc.givenRoot)
|
|
req := httptest.NewRequest(http.MethodGet, tc.whenURL, nil)
|
|
rec := httptest.NewRecorder()
|
|
e.ServeHTTP(rec, req)
|
|
assert.Equal(t, tc.expectStatus, rec.Code)
|
|
body := rec.Body.String()
|
|
if tc.expectBodyStartsWith != "" {
|
|
assert.True(t, strings.HasPrefix(body, tc.expectBodyStartsWith))
|
|
} else {
|
|
assert.Equal(t, "", body)
|
|
}
|
|
|
|
if tc.expectHeaderLocation != "" {
|
|
assert.Equal(t, tc.expectHeaderLocation, rec.Result().Header["Location"][0])
|
|
} else {
|
|
_, ok := rec.Result().Header["Location"]
|
|
assert.False(t, ok)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestEchoStaticRedirectIndex(t *testing.T) {
|
|
e := New()
|
|
|
|
// HandlerFunc
|
|
ri := e.Static("/static", "_fixture")
|
|
assert.Equal(t, http.MethodGet, ri.Method())
|
|
assert.Equal(t, "/static*", ri.Path())
|
|
assert.Equal(t, "GET:/static*", ri.Name())
|
|
assert.Equal(t, []string{"*"}, ri.Params())
|
|
|
|
ctx, cancel := stdContext.WithTimeout(stdContext.Background(), 200*time.Millisecond)
|
|
defer cancel()
|
|
addr, err := startOnRandomPort(ctx, e)
|
|
if err != nil {
|
|
assert.Fail(t, err.Error())
|
|
}
|
|
|
|
code, body, err := doGet(fmt.Sprintf("http://%v/static", addr))
|
|
assert.NoError(t, err)
|
|
assert.True(t, strings.HasPrefix(body, "<!doctype html>"))
|
|
assert.Equal(t, http.StatusOK, code)
|
|
}
|
|
|
|
func TestEchoFile(t *testing.T) {
|
|
e := New()
|
|
ri := e.File("/walle", "_fixture/images/walle.png")
|
|
assert.Equal(t, http.MethodGet, ri.Method())
|
|
assert.Equal(t, "/walle", ri.Path())
|
|
assert.Equal(t, "GET:/walle", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
c, b := request(http.MethodGet, "/walle", e)
|
|
assert.Equal(t, http.StatusOK, c)
|
|
assert.NotEmpty(t, b)
|
|
}
|
|
|
|
func TestEchoMiddleware(t *testing.T) {
|
|
e := New()
|
|
buf := new(bytes.Buffer)
|
|
|
|
e.Pre(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
// before route match is found RouteInfo does not exist
|
|
assert.Equal(t, nil, c.RouteInfo())
|
|
buf.WriteString("-1")
|
|
return next(c)
|
|
}
|
|
})
|
|
|
|
e.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("1")
|
|
return next(c)
|
|
}
|
|
})
|
|
|
|
e.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("2")
|
|
return next(c)
|
|
}
|
|
})
|
|
|
|
e.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("3")
|
|
return next(c)
|
|
}
|
|
})
|
|
|
|
// Route
|
|
e.GET("/", func(c Context) error {
|
|
return c.String(http.StatusOK, "OK")
|
|
})
|
|
|
|
c, b := request(http.MethodGet, "/", e)
|
|
assert.Equal(t, "-1123", buf.String())
|
|
assert.Equal(t, http.StatusOK, c)
|
|
assert.Equal(t, "OK", b)
|
|
}
|
|
|
|
func TestEchoMiddlewareError(t *testing.T) {
|
|
e := New()
|
|
e.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
return errors.New("error")
|
|
}
|
|
})
|
|
e.GET("/", notFoundHandler)
|
|
c, _ := request(http.MethodGet, "/", e)
|
|
assert.Equal(t, http.StatusInternalServerError, c)
|
|
}
|
|
|
|
func TestEchoHandler(t *testing.T) {
|
|
e := New()
|
|
|
|
// HandlerFunc
|
|
e.GET("/ok", func(c Context) error {
|
|
return c.String(http.StatusOK, "OK")
|
|
})
|
|
|
|
c, b := request(http.MethodGet, "/ok", e)
|
|
assert.Equal(t, http.StatusOK, c)
|
|
assert.Equal(t, "OK", b)
|
|
}
|
|
|
|
func TestEchoWrapHandler(t *testing.T) {
|
|
e := New()
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
rec := httptest.NewRecorder()
|
|
c := e.NewContext(req, rec)
|
|
h := WrapHandler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusOK)
|
|
w.Write([]byte("test"))
|
|
}))
|
|
if assert.NoError(t, h(c)) {
|
|
assert.Equal(t, http.StatusOK, rec.Code)
|
|
assert.Equal(t, "test", rec.Body.String())
|
|
}
|
|
}
|
|
|
|
func TestEchoWrapMiddleware(t *testing.T) {
|
|
e := New()
|
|
req := httptest.NewRequest(http.MethodGet, "/", nil)
|
|
rec := httptest.NewRecorder()
|
|
c := e.NewContext(req, rec)
|
|
buf := new(bytes.Buffer)
|
|
mw := WrapMiddleware(func(h http.Handler) http.Handler {
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
buf.Write([]byte("mw"))
|
|
h.ServeHTTP(w, r)
|
|
})
|
|
})
|
|
h := mw(func(c Context) error {
|
|
return c.String(http.StatusOK, "OK")
|
|
})
|
|
if assert.NoError(t, h(c)) {
|
|
assert.Equal(t, "mw", buf.String())
|
|
assert.Equal(t, http.StatusOK, rec.Code)
|
|
assert.Equal(t, "OK", rec.Body.String())
|
|
}
|
|
}
|
|
|
|
func TestEchoGet_routeInfoIsImmutable(t *testing.T) {
|
|
e := New()
|
|
ri := e.GET("/test", handlerFunc)
|
|
assert.Equal(t, "GET:/test", ri.Name())
|
|
|
|
riFromRouter, err := e.Router().Routes().FindByMethodPath(http.MethodGet, "/test")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "GET:/test", riFromRouter.Name())
|
|
|
|
rInfo := ri.(routeInfo)
|
|
rInfo.name = "changed" // this change should not change other returned values
|
|
|
|
assert.Equal(t, "GET:/test", ri.Name())
|
|
|
|
riFromRouter, err = e.Router().Routes().FindByMethodPath(http.MethodGet, "/test")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, "GET:/test", riFromRouter.Name())
|
|
}
|
|
|
|
func TestEchoConnect(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.CONNECT("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodConnect, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodConnect+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodConnect, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoDelete(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.DELETE("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodDelete, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodDelete+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodDelete, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoGet(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.GET("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodGet, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodGet+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodGet, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoHead(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.HEAD("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodHead, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodHead+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodHead, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoOptions(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.OPTIONS("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodOptions, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodOptions+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodOptions, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoPatch(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.PATCH("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodPatch, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodPatch+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodPatch, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoPost(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.POST("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodPost, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodPost+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodPost, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoPut(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.PUT("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodPut, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodPut+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodPut, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoTrace(t *testing.T) {
|
|
e := New()
|
|
|
|
ri := e.TRACE("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
|
|
assert.Equal(t, http.MethodTrace, ri.Method())
|
|
assert.Equal(t, "/", ri.Path())
|
|
assert.Equal(t, http.MethodTrace+":/", ri.Name())
|
|
assert.Nil(t, ri.Params())
|
|
|
|
status, body := request(http.MethodTrace, "/", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "OK", body)
|
|
}
|
|
|
|
func TestEchoAny(t *testing.T) { // JFC
|
|
e := New()
|
|
ris := e.Any("/", func(c Context) error {
|
|
return c.String(http.StatusOK, "Any")
|
|
})
|
|
assert.Len(t, ris, 11)
|
|
}
|
|
|
|
func TestEchoMatch(t *testing.T) { // JFC
|
|
e := New()
|
|
ris := e.Match([]string{http.MethodGet, http.MethodPost}, "/", func(c Context) error {
|
|
return c.String(http.StatusOK, "Match")
|
|
})
|
|
assert.Len(t, ris, 2)
|
|
}
|
|
|
|
func TestEcho_Routers_HandleHostsProperly(t *testing.T) {
|
|
e := New()
|
|
h := e.Host("route.com")
|
|
routes := []*Route{
|
|
{Method: http.MethodGet, Path: "/users/:user/events"},
|
|
{Method: http.MethodGet, Path: "/users/:user/events/public"},
|
|
{Method: http.MethodPost, Path: "/repos/:owner/:repo/git/refs"},
|
|
{Method: http.MethodPost, Path: "/repos/:owner/:repo/git/tags"},
|
|
}
|
|
for _, r := range routes {
|
|
h.Add(r.Method, r.Path, func(c Context) error {
|
|
return c.String(http.StatusOK, "OK")
|
|
})
|
|
}
|
|
|
|
routers := e.Routers()
|
|
|
|
routeCom, ok := routers["route.com"]
|
|
assert.True(t, ok)
|
|
|
|
if assert.Equal(t, len(routes), len(routeCom.Routes())) {
|
|
for _, r := range routeCom.Routes() {
|
|
found := false
|
|
for _, rr := range routes {
|
|
if r.Method() == rr.Method && r.Path() == rr.Path {
|
|
found = true
|
|
break
|
|
}
|
|
}
|
|
if !found {
|
|
t.Errorf("Route %s %s not found", r.Method(), r.Path())
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestEchoServeHTTPPathEncoding(t *testing.T) {
|
|
e := New()
|
|
e.GET("/with/slash", func(c Context) error {
|
|
return c.String(http.StatusOK, "/with/slash")
|
|
})
|
|
e.GET("/:id", func(c Context) error {
|
|
return c.String(http.StatusOK, c.PathParam("id"))
|
|
})
|
|
|
|
var testCases = []struct {
|
|
name string
|
|
whenURL string
|
|
expectURL string
|
|
expectStatus int
|
|
}{
|
|
{
|
|
name: "url with encoding is not decoded for routing",
|
|
whenURL: "/with%2Fslash",
|
|
expectURL: "with%2Fslash", // `%2F` is not decoded to `/` for routing
|
|
expectStatus: http.StatusOK,
|
|
},
|
|
{
|
|
name: "url without encoding is used as is",
|
|
whenURL: "/with/slash",
|
|
expectURL: "/with/slash",
|
|
expectStatus: http.StatusOK,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
req := httptest.NewRequest(http.MethodGet, tc.whenURL, nil)
|
|
rec := httptest.NewRecorder()
|
|
|
|
e.ServeHTTP(rec, req)
|
|
|
|
assert.Equal(t, tc.expectStatus, rec.Code)
|
|
assert.Equal(t, tc.expectURL, rec.Body.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestEchoHost(t *testing.T) {
|
|
okHandler := func(c Context) error { return c.String(http.StatusOK, http.StatusText(http.StatusOK)) }
|
|
teapotHandler := func(c Context) error { return c.String(http.StatusTeapot, http.StatusText(http.StatusTeapot)) }
|
|
acceptHandler := func(c Context) error { return c.String(http.StatusAccepted, http.StatusText(http.StatusAccepted)) }
|
|
teapotMiddleware := MiddlewareFunc(func(next HandlerFunc) HandlerFunc { return teapotHandler })
|
|
|
|
e := New()
|
|
e.GET("/", acceptHandler)
|
|
e.GET("/foo", acceptHandler)
|
|
|
|
ok := e.Host("ok.com")
|
|
ok.GET("/", okHandler)
|
|
ok.GET("/foo", okHandler)
|
|
|
|
teapot := e.Host("teapot.com")
|
|
teapot.GET("/", teapotHandler)
|
|
teapot.GET("/foo", teapotHandler)
|
|
|
|
middle := e.Host("middleware.com", teapotMiddleware)
|
|
middle.GET("/", okHandler)
|
|
middle.GET("/foo", okHandler)
|
|
|
|
var testCases = []struct {
|
|
name string
|
|
whenHost string
|
|
whenPath string
|
|
expectBody string
|
|
expectStatus int
|
|
}{
|
|
{
|
|
name: "No Host Root",
|
|
whenHost: "",
|
|
whenPath: "/",
|
|
expectBody: http.StatusText(http.StatusAccepted),
|
|
expectStatus: http.StatusAccepted,
|
|
},
|
|
{
|
|
name: "No Host Foo",
|
|
whenHost: "",
|
|
whenPath: "/foo",
|
|
expectBody: http.StatusText(http.StatusAccepted),
|
|
expectStatus: http.StatusAccepted,
|
|
},
|
|
{
|
|
name: "OK Host Root",
|
|
whenHost: "ok.com",
|
|
whenPath: "/",
|
|
expectBody: http.StatusText(http.StatusOK),
|
|
expectStatus: http.StatusOK,
|
|
},
|
|
{
|
|
name: "OK Host Foo",
|
|
whenHost: "ok.com",
|
|
whenPath: "/foo",
|
|
expectBody: http.StatusText(http.StatusOK),
|
|
expectStatus: http.StatusOK,
|
|
},
|
|
{
|
|
name: "Teapot Host Root",
|
|
whenHost: "teapot.com",
|
|
whenPath: "/",
|
|
expectBody: http.StatusText(http.StatusTeapot),
|
|
expectStatus: http.StatusTeapot,
|
|
},
|
|
{
|
|
name: "Teapot Host Foo",
|
|
whenHost: "teapot.com",
|
|
whenPath: "/foo",
|
|
expectBody: http.StatusText(http.StatusTeapot),
|
|
expectStatus: http.StatusTeapot,
|
|
},
|
|
{
|
|
name: "Middleware Host",
|
|
whenHost: "middleware.com",
|
|
whenPath: "/",
|
|
expectBody: http.StatusText(http.StatusTeapot),
|
|
expectStatus: http.StatusTeapot,
|
|
},
|
|
{
|
|
name: "Middleware Host Foo",
|
|
whenHost: "middleware.com",
|
|
whenPath: "/foo",
|
|
expectBody: http.StatusText(http.StatusTeapot),
|
|
expectStatus: http.StatusTeapot,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
req := httptest.NewRequest(http.MethodGet, tc.whenPath, nil)
|
|
req.Host = tc.whenHost
|
|
rec := httptest.NewRecorder()
|
|
|
|
e.ServeHTTP(rec, req)
|
|
|
|
assert.Equal(t, tc.expectStatus, rec.Code)
|
|
assert.Equal(t, tc.expectBody, rec.Body.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestEchoGroup(t *testing.T) {
|
|
e := New()
|
|
buf := new(bytes.Buffer)
|
|
e.Use(MiddlewareFunc(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("0")
|
|
return next(c)
|
|
}
|
|
}))
|
|
h := func(c Context) error {
|
|
return c.NoContent(http.StatusOK)
|
|
}
|
|
|
|
//--------
|
|
// Routes
|
|
//--------
|
|
|
|
e.GET("/users", h)
|
|
|
|
// Group
|
|
g1 := e.Group("/group1")
|
|
g1.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("1")
|
|
return next(c)
|
|
}
|
|
})
|
|
g1.GET("", h)
|
|
|
|
// Nested groups with middleware
|
|
g2 := e.Group("/group2")
|
|
g2.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("2")
|
|
return next(c)
|
|
}
|
|
})
|
|
g3 := g2.Group("/group3")
|
|
g3.Use(func(next HandlerFunc) HandlerFunc {
|
|
return func(c Context) error {
|
|
buf.WriteString("3")
|
|
return next(c)
|
|
}
|
|
})
|
|
g3.GET("", h)
|
|
|
|
request(http.MethodGet, "/users", e)
|
|
assert.Equal(t, "0", buf.String())
|
|
|
|
buf.Reset()
|
|
request(http.MethodGet, "/group1", e)
|
|
assert.Equal(t, "01", buf.String())
|
|
|
|
buf.Reset()
|
|
request(http.MethodGet, "/group2/group3", e)
|
|
assert.Equal(t, "023", buf.String())
|
|
}
|
|
|
|
func TestEchoNotFound(t *testing.T) {
|
|
e := New()
|
|
req := httptest.NewRequest(http.MethodGet, "/files", nil)
|
|
rec := httptest.NewRecorder()
|
|
e.ServeHTTP(rec, req)
|
|
assert.Equal(t, http.StatusNotFound, rec.Code)
|
|
}
|
|
|
|
func TestEchoMethodNotAllowed(t *testing.T) {
|
|
e := New()
|
|
e.GET("/", func(c Context) error {
|
|
return c.String(http.StatusOK, "Echo!")
|
|
})
|
|
req := httptest.NewRequest(http.MethodPost, "/", nil)
|
|
rec := httptest.NewRecorder()
|
|
e.ServeHTTP(rec, req)
|
|
assert.Equal(t, http.StatusMethodNotAllowed, rec.Code)
|
|
}
|
|
|
|
func TestEchoContext(t *testing.T) {
|
|
e := New()
|
|
c := e.AcquireContext()
|
|
assert.IsType(t, new(context), c)
|
|
e.ReleaseContext(c)
|
|
}
|
|
|
|
func TestEcho_Start(t *testing.T) {
|
|
e := New()
|
|
e.GET("/", func(c Context) error {
|
|
return c.String(http.StatusTeapot, "OK")
|
|
})
|
|
rndPort, err := net.Listen("tcp", ":0")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer rndPort.Close()
|
|
errChan := make(chan error, 1)
|
|
go func() {
|
|
errChan <- e.Start(rndPort.Addr().String())
|
|
}()
|
|
|
|
select {
|
|
case <-time.After(250 * time.Millisecond):
|
|
t.Fatal("start did not error out")
|
|
case err := <-errChan:
|
|
assert.Contains(t, err.Error(), "bind: address already in use")
|
|
}
|
|
}
|
|
|
|
func request(method, path string, e *Echo) (int, string) {
|
|
req := httptest.NewRequest(method, path, nil)
|
|
rec := httptest.NewRecorder()
|
|
e.ServeHTTP(rec, req)
|
|
return rec.Code, rec.Body.String()
|
|
}
|
|
|
|
func TestDefaultHTTPErrorHandler(t *testing.T) {
|
|
var testCases = []struct {
|
|
name string
|
|
givenExposeError bool
|
|
givenLoggerFunc bool
|
|
whenMethod string
|
|
whenError error
|
|
expectBody string
|
|
expectStatus int
|
|
expectLogged string
|
|
}{
|
|
{
|
|
name: "ok, expose error = true, HTTPError",
|
|
givenExposeError: true,
|
|
whenError: NewHTTPError(http.StatusTeapot, "my_error"),
|
|
expectStatus: http.StatusTeapot,
|
|
expectBody: `{"error":"code=418, message=my_error","message":"my_error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = true, HTTPError + internal error",
|
|
givenExposeError: true,
|
|
whenError: NewHTTPError(http.StatusTeapot, "my_error").WithInternal(errors.New("internal_error")),
|
|
expectStatus: http.StatusTeapot,
|
|
expectBody: `{"error":"code=418, message=my_error, internal=internal_error","message":"my_error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = true, HTTPError + internal HTTPError",
|
|
givenExposeError: true,
|
|
whenError: NewHTTPError(http.StatusTeapot, "my_error").WithInternal(NewHTTPError(http.StatusTooEarly, "early_error")),
|
|
expectStatus: http.StatusTooEarly,
|
|
expectBody: `{"error":"code=418, message=my_error, internal=code=425, message=early_error","message":"early_error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = false, HTTPError",
|
|
whenError: NewHTTPError(http.StatusTeapot, "my_error"),
|
|
expectStatus: http.StatusTeapot,
|
|
expectBody: `{"message":"my_error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = false, HTTPError + internal HTTPError",
|
|
whenError: NewHTTPError(http.StatusTeapot, "my_error").WithInternal(NewHTTPError(http.StatusTooEarly, "early_error")),
|
|
expectStatus: http.StatusTooEarly,
|
|
expectBody: `{"message":"early_error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = true, Error",
|
|
givenExposeError: true,
|
|
whenError: fmt.Errorf("my errors wraps: %w", errors.New("internal_error")),
|
|
expectStatus: http.StatusInternalServerError,
|
|
expectBody: `{"error":"my errors wraps: internal_error","message":"Internal Server Error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, expose error = false, Error",
|
|
whenError: fmt.Errorf("my errors wraps: %w", errors.New("internal_error")),
|
|
expectStatus: http.StatusInternalServerError,
|
|
expectBody: `{"message":"Internal Server Error"}` + "\n",
|
|
},
|
|
{
|
|
name: "ok, http.HEAD, expose error = true, Error",
|
|
givenExposeError: true,
|
|
whenMethod: http.MethodHead,
|
|
whenError: fmt.Errorf("my errors wraps: %w", errors.New("internal_error")),
|
|
expectStatus: http.StatusInternalServerError,
|
|
expectBody: ``,
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.name, func(t *testing.T) {
|
|
buf := new(bytes.Buffer)
|
|
e := New()
|
|
e.Logger = &jsonLogger{writer: buf}
|
|
e.Any("/path", func(c Context) error {
|
|
return tc.whenError
|
|
})
|
|
|
|
e.HTTPErrorHandler = DefaultHTTPErrorHandler(tc.givenExposeError)
|
|
|
|
method := http.MethodGet
|
|
if tc.whenMethod != "" {
|
|
method = tc.whenMethod
|
|
}
|
|
c, b := request(method, "/path", e)
|
|
|
|
assert.Equal(t, tc.expectStatus, c)
|
|
assert.Equal(t, tc.expectBody, b)
|
|
assert.Equal(t, tc.expectLogged, buf.String())
|
|
})
|
|
}
|
|
}
|
|
|
|
type myCustomContext struct {
|
|
context
|
|
}
|
|
|
|
func (c *myCustomContext) QueryParam(name string) string {
|
|
return "prefix_" + name
|
|
}
|
|
|
|
func TestEcho_customContext(t *testing.T) {
|
|
e := New()
|
|
e.NewContextFunc = func(pathParamAllocSize int) EditableContext {
|
|
p := make(PathParams, pathParamAllocSize)
|
|
return &myCustomContext{
|
|
context{
|
|
request: nil,
|
|
response: NewResponse(nil, e),
|
|
store: make(Map),
|
|
echo: e,
|
|
pathParams: &p,
|
|
route: nil,
|
|
},
|
|
}
|
|
}
|
|
|
|
e.GET("/info/:id/:file", func(c Context) error {
|
|
return c.String(http.StatusTeapot, c.QueryParam("param"))
|
|
})
|
|
|
|
status, body := request(http.MethodGet, "/info/1/a.csv", e)
|
|
assert.Equal(t, http.StatusTeapot, status)
|
|
assert.Equal(t, "prefix_param", body)
|
|
}
|
|
|
|
func benchmarkEchoRoutes(b *testing.B, routes []testRoute) {
|
|
e := New()
|
|
req := httptest.NewRequest("GET", "/", nil)
|
|
u := req.URL
|
|
w := httptest.NewRecorder()
|
|
|
|
b.ReportAllocs()
|
|
|
|
// Add routes
|
|
for _, route := range routes {
|
|
e.Add(route.Method, route.Path, func(c Context) error {
|
|
return nil
|
|
})
|
|
}
|
|
|
|
// Find routes
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
for _, route := range routes {
|
|
req.Method = route.Method
|
|
u.Path = route.Path
|
|
e.ServeHTTP(w, req)
|
|
}
|
|
}
|
|
}
|
|
|
|
func BenchmarkEchoStaticRoutes(b *testing.B) {
|
|
benchmarkEchoRoutes(b, staticRoutes)
|
|
}
|
|
|
|
func BenchmarkEchoStaticRoutesMisses(b *testing.B) {
|
|
benchmarkEchoRoutes(b, staticRoutes)
|
|
}
|
|
|
|
func BenchmarkEchoGitHubAPI(b *testing.B) {
|
|
benchmarkEchoRoutes(b, gitHubAPI)
|
|
}
|
|
|
|
func BenchmarkEchoGitHubAPIMisses(b *testing.B) {
|
|
benchmarkEchoRoutes(b, gitHubAPI)
|
|
}
|
|
|
|
func BenchmarkEchoParseAPI(b *testing.B) {
|
|
benchmarkEchoRoutes(b, parseAPI)
|
|
}
|