mirror of
				https://github.com/gin-gonic/gin.git
				synced 2025-10-25 03:12:16 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			464 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			464 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2014 Manu Martinez-Almeida.  All rights reserved.
 | |
| // Use of this source code is governed by a MIT style
 | |
| // license that can be found in the LICENSE file.
 | |
| 
 | |
| package gin
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"net/http"
 | |
| 	"net/http/httptest"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| func performRequest(r http.Handler, method, path string) *httptest.ResponseRecorder {
 | |
| 	req, _ := http.NewRequest(method, path, nil)
 | |
| 	w := httptest.NewRecorder()
 | |
| 	r.ServeHTTP(w, req)
 | |
| 	return w
 | |
| }
 | |
| 
 | |
| func testRouteOK(method string, t *testing.T) {
 | |
| 	passed := false
 | |
| 	passedAny := false
 | |
| 	r := New()
 | |
| 	r.Any("/test2", func(c *Context) {
 | |
| 		passedAny = true
 | |
| 	})
 | |
| 	r.Handle(method, "/test", func(c *Context) {
 | |
| 		passed = true
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(r, method, "/test")
 | |
| 	assert.True(t, passed)
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 
 | |
| 	performRequest(r, method, "/test2")
 | |
| 	assert.True(t, passedAny)
 | |
| }
 | |
| 
 | |
| // TestSingleRouteOK tests that POST route is correctly invoked.
 | |
| func testRouteNotOK(method string, t *testing.T) {
 | |
| 	passed := false
 | |
| 	router := New()
 | |
| 	router.Handle(method, "/test_2", func(c *Context) {
 | |
| 		passed = true
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(router, method, "/test")
 | |
| 
 | |
| 	assert.False(t, passed)
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| }
 | |
| 
 | |
| // TestSingleRouteOK tests that POST route is correctly invoked.
 | |
| func testRouteNotOK2(method string, t *testing.T) {
 | |
| 	passed := false
 | |
| 	router := New()
 | |
| 	router.HandleMethodNotAllowed = true
 | |
| 	var methodRoute string
 | |
| 	if method == "POST" {
 | |
| 		methodRoute = "GET"
 | |
| 	} else {
 | |
| 		methodRoute = "POST"
 | |
| 	}
 | |
| 	router.Handle(methodRoute, "/test", func(c *Context) {
 | |
| 		passed = true
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(router, method, "/test")
 | |
| 
 | |
| 	assert.False(t, passed)
 | |
| 	assert.Equal(t, http.StatusMethodNotAllowed, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouterMethod(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.PUT("/hey2", func(c *Context) {
 | |
| 		c.String(http.StatusOK, "sup2")
 | |
| 	})
 | |
| 
 | |
| 	router.PUT("/hey", func(c *Context) {
 | |
| 		c.String(http.StatusOK, "called")
 | |
| 	})
 | |
| 
 | |
| 	router.PUT("/hey3", func(c *Context) {
 | |
| 		c.String(http.StatusOK, "sup3")
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(router, "PUT", "/hey")
 | |
| 
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 	assert.Equal(t, "called", w.Body.String())
 | |
| }
 | |
| 
 | |
| func TestRouterGroupRouteOK(t *testing.T) {
 | |
| 	testRouteOK("GET", t)
 | |
| 	testRouteOK("POST", t)
 | |
| 	testRouteOK("PUT", t)
 | |
| 	testRouteOK("PATCH", t)
 | |
| 	testRouteOK("HEAD", t)
 | |
| 	testRouteOK("OPTIONS", t)
 | |
| 	testRouteOK("DELETE", t)
 | |
| 	testRouteOK("CONNECT", t)
 | |
| 	testRouteOK("TRACE", t)
 | |
| }
 | |
| 
 | |
| func TestRouteNotOK(t *testing.T) {
 | |
| 	testRouteNotOK("GET", t)
 | |
| 	testRouteNotOK("POST", t)
 | |
| 	testRouteNotOK("PUT", t)
 | |
| 	testRouteNotOK("PATCH", t)
 | |
| 	testRouteNotOK("HEAD", t)
 | |
| 	testRouteNotOK("OPTIONS", t)
 | |
| 	testRouteNotOK("DELETE", t)
 | |
| 	testRouteNotOK("CONNECT", t)
 | |
| 	testRouteNotOK("TRACE", t)
 | |
| }
 | |
| 
 | |
| func TestRouteNotOK2(t *testing.T) {
 | |
| 	testRouteNotOK2("GET", t)
 | |
| 	testRouteNotOK2("POST", t)
 | |
| 	testRouteNotOK2("PUT", t)
 | |
| 	testRouteNotOK2("PATCH", t)
 | |
| 	testRouteNotOK2("HEAD", t)
 | |
| 	testRouteNotOK2("OPTIONS", t)
 | |
| 	testRouteNotOK2("DELETE", t)
 | |
| 	testRouteNotOK2("CONNECT", t)
 | |
| 	testRouteNotOK2("TRACE", t)
 | |
| }
 | |
| 
 | |
| func TestRouteRedirectTrailingSlash(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.RedirectFixedPath = false
 | |
| 	router.RedirectTrailingSlash = true
 | |
| 	router.GET("/path", func(c *Context) {})
 | |
| 	router.GET("/path2/", func(c *Context) {})
 | |
| 	router.POST("/path3", func(c *Context) {})
 | |
| 	router.PUT("/path4/", func(c *Context) {})
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/path/")
 | |
| 	assert.Equal(t, "/path", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusMovedPermanently, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "GET", "/path2")
 | |
| 	assert.Equal(t, "/path2/", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusMovedPermanently, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "POST", "/path3/")
 | |
| 	assert.Equal(t, "/path3", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusTemporaryRedirect, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "PUT", "/path4")
 | |
| 	assert.Equal(t, "/path4/", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusTemporaryRedirect, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "GET", "/path")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "GET", "/path2/")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "POST", "/path3")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "PUT", "/path4/")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 
 | |
| 	router.RedirectTrailingSlash = false
 | |
| 
 | |
| 	w = performRequest(router, "GET", "/path/")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 	w = performRequest(router, "GET", "/path2")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 	w = performRequest(router, "POST", "/path3/")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 	w = performRequest(router, "PUT", "/path4")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteRedirectFixedPath(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.RedirectFixedPath = true
 | |
| 	router.RedirectTrailingSlash = false
 | |
| 
 | |
| 	router.GET("/path", func(c *Context) {})
 | |
| 	router.GET("/Path2", func(c *Context) {})
 | |
| 	router.POST("/PATH3", func(c *Context) {})
 | |
| 	router.POST("/Path4/", func(c *Context) {})
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/PATH")
 | |
| 	assert.Equal(t, "/path", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusMovedPermanently, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "GET", "/path2")
 | |
| 	assert.Equal(t, "/Path2", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusMovedPermanently, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "POST", "/path3")
 | |
| 	assert.Equal(t, "/PATH3", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusTemporaryRedirect, w.Code)
 | |
| 
 | |
| 	w = performRequest(router, "POST", "/path4")
 | |
| 	assert.Equal(t, "/Path4/", w.Header().Get("Location"))
 | |
| 	assert.Equal(t, http.StatusTemporaryRedirect, w.Code)
 | |
| }
 | |
| 
 | |
| // TestContextParamsGet tests that a parameter can be parsed from the URL.
 | |
| func TestRouteParamsByName(t *testing.T) {
 | |
| 	name := ""
 | |
| 	lastName := ""
 | |
| 	wild := ""
 | |
| 	router := New()
 | |
| 	router.GET("/test/:name/:last_name/*wild", func(c *Context) {
 | |
| 		name = c.Params.ByName("name")
 | |
| 		lastName = c.Params.ByName("last_name")
 | |
| 		var ok bool
 | |
| 		wild, ok = c.Params.Get("wild")
 | |
| 
 | |
| 		assert.True(t, ok)
 | |
| 		assert.Equal(t, name, c.Param("name"))
 | |
| 		assert.Equal(t, name, c.Param("name"))
 | |
| 		assert.Equal(t, lastName, c.Param("last_name"))
 | |
| 
 | |
| 		assert.Empty(t, c.Param("wtf"))
 | |
| 		assert.Empty(t, c.Params.ByName("wtf"))
 | |
| 
 | |
| 		wtf, ok := c.Params.Get("wtf")
 | |
| 		assert.Empty(t, wtf)
 | |
| 		assert.False(t, ok)
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/test/john/smith/is/super/great")
 | |
| 
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 	assert.Equal(t, "john", name)
 | |
| 	assert.Equal(t, "smith", lastName)
 | |
| 	assert.Equal(t, "/is/super/great", wild)
 | |
| }
 | |
| 
 | |
| // TestHandleStaticFile - ensure the static file handles properly
 | |
| func TestRouteStaticFile(t *testing.T) {
 | |
| 	// SETUP file
 | |
| 	testRoot, _ := os.Getwd()
 | |
| 	f, err := ioutil.TempFile(testRoot, "")
 | |
| 	if err != nil {
 | |
| 		t.Error(err)
 | |
| 	}
 | |
| 	defer os.Remove(f.Name())
 | |
| 	f.WriteString("Gin Web Framework")
 | |
| 	f.Close()
 | |
| 
 | |
| 	dir, filename := filepath.Split(f.Name())
 | |
| 
 | |
| 	// SETUP gin
 | |
| 	router := New()
 | |
| 	router.Static("/using_static", dir)
 | |
| 	router.StaticFile("/result", f.Name())
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/using_static/"+filename)
 | |
| 	w2 := performRequest(router, "GET", "/result")
 | |
| 
 | |
| 	assert.Equal(t, w, w2)
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 	assert.Equal(t, "Gin Web Framework", w.Body.String())
 | |
| 	assert.Equal(t, "text/plain; charset=utf-8", w.HeaderMap.Get("Content-Type"))
 | |
| 
 | |
| 	w3 := performRequest(router, "HEAD", "/using_static/"+filename)
 | |
| 	w4 := performRequest(router, "HEAD", "/result")
 | |
| 
 | |
| 	assert.Equal(t, w3, w4)
 | |
| 	assert.Equal(t, http.StatusOK, w3.Code)
 | |
| }
 | |
| 
 | |
| // TestHandleStaticDir - ensure the root/sub dir handles properly
 | |
| func TestRouteStaticListingDir(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.StaticFS("/", Dir("./", true))
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/")
 | |
| 
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 	assert.Contains(t, w.Body.String(), "gin.go")
 | |
| 	assert.Equal(t, "text/html; charset=utf-8", w.HeaderMap.Get("Content-Type"))
 | |
| }
 | |
| 
 | |
| // TestHandleHeadToDir - ensure the root/sub dir handles properly
 | |
| func TestRouteStaticNoListing(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.Static("/", "./")
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/")
 | |
| 
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 	assert.NotContains(t, w.Body.String(), "gin.go")
 | |
| }
 | |
| 
 | |
| func TestRouterMiddlewareAndStatic(t *testing.T) {
 | |
| 	router := New()
 | |
| 	static := router.Group("/", func(c *Context) {
 | |
| 		c.Writer.Header().Add("Last-Modified", "Mon, 02 Jan 2006 15:04:05 MST")
 | |
| 		c.Writer.Header().Add("Expires", "Mon, 02 Jan 2006 15:04:05 MST")
 | |
| 		c.Writer.Header().Add("X-GIN", "Gin Framework")
 | |
| 	})
 | |
| 	static.Static("/", "./")
 | |
| 
 | |
| 	w := performRequest(router, "GET", "/gin.go")
 | |
| 
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| 	assert.Contains(t, w.Body.String(), "package gin")
 | |
| 	assert.Equal(t, "text/plain; charset=utf-8", w.HeaderMap.Get("Content-Type"))
 | |
| 	assert.NotEqual(t, w.HeaderMap.Get("Last-Modified"), "Mon, 02 Jan 2006 15:04:05 MST")
 | |
| 	assert.Equal(t, "Mon, 02 Jan 2006 15:04:05 MST", w.HeaderMap.Get("Expires"))
 | |
| 	assert.Equal(t, "Gin Framework", w.HeaderMap.Get("x-GIN"))
 | |
| }
 | |
| 
 | |
| func TestRouteNotAllowedEnabled(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.HandleMethodNotAllowed = true
 | |
| 	router.POST("/path", func(c *Context) {})
 | |
| 	w := performRequest(router, "GET", "/path")
 | |
| 	assert.Equal(t, http.StatusMethodNotAllowed, w.Code)
 | |
| 
 | |
| 	router.NoMethod(func(c *Context) {
 | |
| 		c.String(http.StatusTeapot, "responseText")
 | |
| 	})
 | |
| 	w = performRequest(router, "GET", "/path")
 | |
| 	assert.Equal(t, "responseText", w.Body.String())
 | |
| 	assert.Equal(t, http.StatusTeapot, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteNotAllowedEnabled2(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.HandleMethodNotAllowed = true
 | |
| 	// add one methodTree to trees
 | |
| 	router.addRoute("POST", "/", HandlersChain{func(_ *Context) {}})
 | |
| 	router.GET("/path2", func(c *Context) {})
 | |
| 	w := performRequest(router, "POST", "/path2")
 | |
| 	assert.Equal(t, http.StatusMethodNotAllowed, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteNotAllowedDisabled(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.HandleMethodNotAllowed = false
 | |
| 	router.POST("/path", func(c *Context) {})
 | |
| 	w := performRequest(router, "GET", "/path")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 
 | |
| 	router.NoMethod(func(c *Context) {
 | |
| 		c.String(http.StatusTeapot, "responseText")
 | |
| 	})
 | |
| 	w = performRequest(router, "GET", "/path")
 | |
| 	assert.Equal(t, "404 page not found", w.Body.String())
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouterNotFound(t *testing.T) {
 | |
| 	router := New()
 | |
| 	router.RedirectFixedPath = true
 | |
| 	router.GET("/path", func(c *Context) {})
 | |
| 	router.GET("/dir/", func(c *Context) {})
 | |
| 	router.GET("/", func(c *Context) {})
 | |
| 
 | |
| 	testRoutes := []struct {
 | |
| 		route    string
 | |
| 		code     int
 | |
| 		location string
 | |
| 	}{
 | |
| 		{"/path/", http.StatusMovedPermanently, "/path"},   // TSR -/
 | |
| 		{"/dir", http.StatusMovedPermanently, "/dir/"},     // TSR +/
 | |
| 		{"", http.StatusMovedPermanently, "/"},             // TSR +/
 | |
| 		{"/PATH", http.StatusMovedPermanently, "/path"},    // Fixed Case
 | |
| 		{"/DIR/", http.StatusMovedPermanently, "/dir/"},    // Fixed Case
 | |
| 		{"/PATH/", http.StatusMovedPermanently, "/path"},   // Fixed Case -/
 | |
| 		{"/DIR", http.StatusMovedPermanently, "/dir/"},     // Fixed Case +/
 | |
| 		{"/../path", http.StatusMovedPermanently, "/path"}, // CleanPath
 | |
| 		{"/nope", http.StatusNotFound, ""},                 // NotFound
 | |
| 	}
 | |
| 	for _, tr := range testRoutes {
 | |
| 		w := performRequest(router, "GET", tr.route)
 | |
| 		assert.Equal(t, tr.code, w.Code)
 | |
| 		if w.Code != http.StatusNotFound {
 | |
| 			assert.Equal(t, tr.location, fmt.Sprint(w.Header().Get("Location")))
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	// Test custom not found handler
 | |
| 	var notFound bool
 | |
| 	router.NoRoute(func(c *Context) {
 | |
| 		c.AbortWithStatus(http.StatusNotFound)
 | |
| 		notFound = true
 | |
| 	})
 | |
| 	w := performRequest(router, "GET", "/nope")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| 	assert.True(t, notFound)
 | |
| 
 | |
| 	// Test other method than GET (want 307 instead of 301)
 | |
| 	router.PATCH("/path", func(c *Context) {})
 | |
| 	w = performRequest(router, "PATCH", "/path/")
 | |
| 	assert.Equal(t, http.StatusTemporaryRedirect, w.Code)
 | |
| 	assert.Equal(t, "map[Location:[/path]]", fmt.Sprint(w.Header()))
 | |
| 
 | |
| 	// Test special case where no node for the prefix "/" exists
 | |
| 	router = New()
 | |
| 	router.GET("/a", func(c *Context) {})
 | |
| 	w = performRequest(router, "GET", "/")
 | |
| 	assert.Equal(t, http.StatusNotFound, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteRawPath(t *testing.T) {
 | |
| 	route := New()
 | |
| 	route.UseRawPath = true
 | |
| 
 | |
| 	route.POST("/project/:name/build/:num", func(c *Context) {
 | |
| 		name := c.Params.ByName("name")
 | |
| 		num := c.Params.ByName("num")
 | |
| 
 | |
| 		assert.Equal(t, name, c.Param("name"))
 | |
| 		assert.Equal(t, num, c.Param("num"))
 | |
| 
 | |
| 		assert.Equal(t, "Some/Other/Project", name)
 | |
| 		assert.Equal(t, "222", num)
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(route, "POST", "/project/Some%2FOther%2FProject/build/222")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteRawPathNoUnescape(t *testing.T) {
 | |
| 	route := New()
 | |
| 	route.UseRawPath = true
 | |
| 	route.UnescapePathValues = false
 | |
| 
 | |
| 	route.POST("/project/:name/build/:num", func(c *Context) {
 | |
| 		name := c.Params.ByName("name")
 | |
| 		num := c.Params.ByName("num")
 | |
| 
 | |
| 		assert.Equal(t, name, c.Param("name"))
 | |
| 		assert.Equal(t, num, c.Param("num"))
 | |
| 
 | |
| 		assert.Equal(t, "Some%2FOther%2FProject", name)
 | |
| 		assert.Equal(t, "333", num)
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(route, "POST", "/project/Some%2FOther%2FProject/build/333")
 | |
| 	assert.Equal(t, http.StatusOK, w.Code)
 | |
| }
 | |
| 
 | |
| func TestRouteServeErrorWithWriteHeader(t *testing.T) {
 | |
| 	route := New()
 | |
| 	route.Use(func(c *Context) {
 | |
| 		c.Status(421)
 | |
| 		c.Next()
 | |
| 	})
 | |
| 
 | |
| 	w := performRequest(route, "GET", "/NotFound")
 | |
| 	assert.Equal(t, 421, w.Code)
 | |
| 	assert.Equal(t, 0, w.Body.Len())
 | |
| }
 |