2014-08-29 21:49:50 +04:00
|
|
|
// 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.
|
|
|
|
|
2014-07-04 12:12:28 +04:00
|
|
|
package gin
|
|
|
|
|
2014-07-05 19:04:11 +04:00
|
|
|
import (
|
2017-11-21 04:27:57 +03:00
|
|
|
"crypto/tls"
|
2017-05-29 11:03:49 +03:00
|
|
|
"fmt"
|
2017-07-02 07:38:05 +03:00
|
|
|
"html/template"
|
2017-05-29 11:03:49 +03:00
|
|
|
"io/ioutil"
|
2021-04-06 06:37:25 +03:00
|
|
|
"net"
|
2017-05-29 11:03:49 +03:00
|
|
|
"net/http"
|
2018-10-16 03:48:41 +03:00
|
|
|
"net/http/httptest"
|
2015-05-31 23:35:49 +03:00
|
|
|
"reflect"
|
2019-02-18 04:35:08 +03:00
|
|
|
"strconv"
|
|
|
|
"sync/atomic"
|
2014-07-05 19:04:11 +04:00
|
|
|
"testing"
|
2017-05-29 11:03:49 +03:00
|
|
|
"time"
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
2014-07-04 12:12:28 +04:00
|
|
|
)
|
|
|
|
|
2017-07-02 07:38:05 +03:00
|
|
|
func formatAsDate(t time.Time) string {
|
|
|
|
year, month, day := t.Date()
|
|
|
|
return fmt.Sprintf("%d/%02d/%02d", year, month, day)
|
|
|
|
}
|
|
|
|
|
2018-10-16 03:48:41 +03:00
|
|
|
func setupHTMLFiles(t *testing.T, mode string, tls bool, loadMethod func(*Engine)) *httptest.Server {
|
|
|
|
SetMode(mode)
|
2019-02-18 05:10:45 +03:00
|
|
|
defer SetMode(TestMode)
|
|
|
|
|
|
|
|
var router *Engine
|
|
|
|
captureOutput(t, func() {
|
|
|
|
router = New()
|
|
|
|
router.Delims("{[{", "}]}")
|
|
|
|
router.SetFuncMap(template.FuncMap{
|
|
|
|
"formatAsDate": formatAsDate,
|
|
|
|
})
|
|
|
|
loadMethod(router)
|
|
|
|
router.GET("/test", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"})
|
|
|
|
})
|
|
|
|
router.GET("/raw", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{
|
|
|
|
"now": time.Date(2017, 07, 01, 0, 0, 0, 0, time.UTC),
|
|
|
|
})
|
2017-07-02 08:06:43 +03:00
|
|
|
})
|
2018-10-16 03:48:41 +03:00
|
|
|
})
|
2017-05-29 11:03:49 +03:00
|
|
|
|
2018-10-16 03:48:41 +03:00
|
|
|
var ts *httptest.Server
|
|
|
|
|
|
|
|
if tls {
|
|
|
|
ts = httptest.NewTLSServer(router)
|
|
|
|
} else {
|
|
|
|
ts = httptest.NewServer(router)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ts
|
2017-05-29 11:03:49 +03:00
|
|
|
}
|
|
|
|
|
2017-11-21 04:27:57 +03:00
|
|
|
func setupHTMLFiles(t *testing.T, mode string, tls bool) func() {
|
2017-05-29 11:03:49 +03:00
|
|
|
go func() {
|
2017-11-21 04:27:57 +03:00
|
|
|
SetMode(mode)
|
2017-05-29 11:03:49 +03:00
|
|
|
router := New()
|
|
|
|
router.Delims("{[{", "}]}")
|
2017-07-02 08:06:43 +03:00
|
|
|
router.SetFuncMap(template.FuncMap{
|
|
|
|
"formatAsDate": formatAsDate,
|
|
|
|
})
|
|
|
|
router.LoadHTMLFiles("./fixtures/basic/hello.tmpl", "./fixtures/basic/raw.tmpl")
|
2017-05-29 11:03:49 +03:00
|
|
|
router.GET("/test", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"})
|
|
|
|
})
|
2017-07-02 08:06:43 +03:00
|
|
|
router.GET("/raw", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{
|
|
|
|
"now": time.Date(2017, 07, 01, 0, 0, 0, 0, time.UTC),
|
|
|
|
})
|
|
|
|
})
|
2017-11-21 04:27:57 +03:00
|
|
|
if tls {
|
|
|
|
// these files generated by `go run $GOROOT/src/crypto/tls/generate_cert.go --host 127.0.0.1`
|
|
|
|
router.RunTLS(":9999", "./fixtures/testdata/cert.pem", "./fixtures/testdata/key.pem")
|
|
|
|
} else {
|
|
|
|
router.Run(":8888")
|
|
|
|
}
|
2017-05-29 11:03:49 +03:00
|
|
|
}()
|
|
|
|
t.Log("waiting 1 second for server startup")
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
return func() {}
|
|
|
|
}
|
2018-10-16 03:48:41 +03:00
|
|
|
|
2017-11-21 04:27:57 +03:00
|
|
|
func setupHTMLGlob(t *testing.T, mode string, tls bool) func() {
|
2017-05-29 11:03:49 +03:00
|
|
|
go func() {
|
2017-11-21 04:27:57 +03:00
|
|
|
SetMode(mode)
|
2017-05-29 11:03:49 +03:00
|
|
|
router := New()
|
|
|
|
router.Delims("{[{", "}]}")
|
2017-07-02 07:38:05 +03:00
|
|
|
router.SetFuncMap(template.FuncMap{
|
|
|
|
"formatAsDate": formatAsDate,
|
|
|
|
})
|
2017-05-29 11:03:49 +03:00
|
|
|
router.LoadHTMLGlob("./fixtures/basic/*")
|
|
|
|
router.GET("/test", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"})
|
|
|
|
})
|
2017-07-02 07:38:05 +03:00
|
|
|
router.GET("/raw", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{
|
|
|
|
"now": time.Date(2017, 07, 01, 0, 0, 0, 0, time.UTC),
|
|
|
|
})
|
|
|
|
})
|
2017-11-21 04:27:57 +03:00
|
|
|
if tls {
|
|
|
|
// these files generated by `go run $GOROOT/src/crypto/tls/generate_cert.go --host 127.0.0.1`
|
|
|
|
router.RunTLS(":9999", "./fixtures/testdata/cert.pem", "./fixtures/testdata/key.pem")
|
|
|
|
} else {
|
|
|
|
router.Run(":8888")
|
|
|
|
}
|
2017-05-29 11:03:49 +03:00
|
|
|
}()
|
|
|
|
t.Log("waiting 1 second for server startup")
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
return func() {}
|
|
|
|
}
|
|
|
|
|
2018-03-23 01:51:42 +03:00
|
|
|
func setupHTMLFilesRecursively(t *testing.T, mode string, tls bool) func() {
|
|
|
|
go func() {
|
|
|
|
SetMode(mode)
|
|
|
|
router := New()
|
|
|
|
router.Delims("{[{", "}]}")
|
|
|
|
router.SetFuncMap(template.FuncMap{
|
|
|
|
"formatAsDate": formatAsDate,
|
|
|
|
})
|
|
|
|
router.LoadHTMLFilesRecursively("./fixtures/basic/", []string{".tmpl"})
|
|
|
|
router.GET("/test", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"})
|
|
|
|
})
|
|
|
|
router.GET("/raw", func(c *Context) {
|
|
|
|
c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{
|
|
|
|
"now": time.Date(2017, 07, 01, 0, 0, 0, 0, time.UTC),
|
|
|
|
})
|
|
|
|
})
|
|
|
|
if tls {
|
|
|
|
// these files generated by `go run $GOROOT/src/crypto/tls/generate_cert.go --host 127.0.0.1`
|
|
|
|
router.RunTLS(":9999", "./fixtures/testdata/cert.pem", "./fixtures/testdata/key.pem")
|
|
|
|
} else {
|
|
|
|
router.Run(":8888")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
t.Log("waiting 1 second for server startup")
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
return func() {}
|
|
|
|
}
|
|
|
|
|
2017-05-29 11:03:49 +03:00
|
|
|
func TestLoadHTMLGlob(t *testing.T) {
|
2017-11-21 04:27:57 +03:00
|
|
|
td := setupHTMLGlob(t, DebugMode, false)
|
2017-05-29 11:03:49 +03:00
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
2018-10-16 03:48:41 +03:00
|
|
|
|
2017-11-21 04:27:57 +03:00
|
|
|
func TestLoadHTMLGlob2(t *testing.T) {
|
|
|
|
td := setupHTMLGlob(t, TestMode, false)
|
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
2018-10-16 03:48:41 +03:00
|
|
|
|
2017-11-21 04:27:57 +03:00
|
|
|
func TestLoadHTMLGlob3(t *testing.T) {
|
|
|
|
td := setupHTMLGlob(t, ReleaseMode, false)
|
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoadHTMLGlobUsingTLS(t *testing.T) {
|
|
|
|
td := setupHTMLGlob(t, DebugMode, true)
|
|
|
|
// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error
|
|
|
|
tr := &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
client := &http.Client{Transport: tr}
|
|
|
|
res, err := client.Get("https://127.0.0.1:9999/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
2017-07-02 08:06:43 +03:00
|
|
|
func TestLoadHTMLGlobFromFuncMap(t *testing.T) {
|
2017-07-02 07:38:05 +03:00
|
|
|
time.Now()
|
2017-11-21 04:27:57 +03:00
|
|
|
td := setupHTMLGlob(t, DebugMode, false)
|
2017-07-02 07:38:05 +03:00
|
|
|
res, err := http.Get("http://127.0.0.1:8888/raw")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "Date: 2017/07/01\n", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
2014-08-21 03:01:42 +04:00
|
|
|
func init() {
|
|
|
|
SetMode(TestMode)
|
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestCreateEngine(t *testing.T) {
|
|
|
|
router := New()
|
2015-07-08 05:27:23 +03:00
|
|
|
assert.Equal(t, "/", router.basePath)
|
2015-04-08 03:58:35 +03:00
|
|
|
assert.Equal(t, router.engine, router)
|
|
|
|
assert.Empty(t, router.Handlers)
|
2015-06-04 05:32:18 +03:00
|
|
|
}
|
|
|
|
|
2018-10-16 03:48:41 +03:00
|
|
|
func TestLoadHTMLFilesTestMode(t *testing.T) {
|
|
|
|
ts := setupHTMLFiles(
|
|
|
|
t,
|
|
|
|
TestMode,
|
|
|
|
false,
|
|
|
|
func(router *Engine) {
|
|
|
|
router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
res, err := http.Get(fmt.Sprintf("%s/test", ts.URL))
|
2017-05-29 11:03:49 +03:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
2018-09-12 16:21:26 +03:00
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp))
|
2017-05-29 11:03:49 +03:00
|
|
|
}
|
|
|
|
|
2018-10-16 03:48:41 +03:00
|
|
|
func TestLoadHTMLFilesDebugMode(t *testing.T) {
|
|
|
|
ts := setupHTMLFiles(
|
|
|
|
t,
|
|
|
|
DebugMode,
|
|
|
|
false,
|
|
|
|
func(router *Engine) {
|
|
|
|
router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
res, err := http.Get(fmt.Sprintf("%s/test", ts.URL))
|
2017-11-21 04:27:57 +03:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
2018-09-12 16:21:26 +03:00
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp))
|
2017-11-21 04:27:57 +03:00
|
|
|
}
|
|
|
|
|
2018-10-16 03:48:41 +03:00
|
|
|
func TestLoadHTMLFilesReleaseMode(t *testing.T) {
|
|
|
|
ts := setupHTMLFiles(
|
|
|
|
t,
|
|
|
|
ReleaseMode,
|
|
|
|
false,
|
|
|
|
func(router *Engine) {
|
|
|
|
router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
res, err := http.Get(fmt.Sprintf("%s/test", ts.URL))
|
2017-11-21 04:27:57 +03:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
2018-09-12 16:21:26 +03:00
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp))
|
2017-11-21 04:27:57 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoadHTMLFilesUsingTLS(t *testing.T) {
|
2018-10-16 03:48:41 +03:00
|
|
|
ts := setupHTMLFiles(
|
|
|
|
t,
|
|
|
|
TestMode,
|
|
|
|
true,
|
|
|
|
func(router *Engine) {
|
|
|
|
router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
|
2017-11-21 04:27:57 +03:00
|
|
|
// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error
|
|
|
|
tr := &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
client := &http.Client{Transport: tr}
|
2018-10-16 03:48:41 +03:00
|
|
|
res, err := client.Get(fmt.Sprintf("%s/test", ts.URL))
|
2017-11-21 04:27:57 +03:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
2018-09-12 16:21:26 +03:00
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp))
|
2017-11-21 04:27:57 +03:00
|
|
|
}
|
|
|
|
|
2017-07-02 08:06:43 +03:00
|
|
|
func TestLoadHTMLFilesFuncMap(t *testing.T) {
|
2018-10-16 03:48:41 +03:00
|
|
|
ts := setupHTMLFiles(
|
|
|
|
t,
|
|
|
|
TestMode,
|
|
|
|
false,
|
|
|
|
func(router *Engine) {
|
|
|
|
router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl")
|
|
|
|
},
|
|
|
|
)
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
res, err := http.Get(fmt.Sprintf("%s/raw", ts.URL))
|
2017-07-02 08:06:43 +03:00
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
2018-09-12 16:21:26 +03:00
|
|
|
assert.Equal(t, "Date: 2017/07/01\n", string(resp))
|
2017-07-02 08:06:43 +03:00
|
|
|
}
|
|
|
|
|
2018-03-23 01:51:42 +03:00
|
|
|
func TestLoadHTMLFilesRecursively(t *testing.T) {
|
2018-03-23 21:14:34 +03:00
|
|
|
td := setupHTMLFilesRecursively(t, TestMode, false)
|
2018-03-23 01:51:42 +03:00
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoadHTMLFilesRecursively2(t *testing.T) {
|
2018-03-23 21:14:34 +03:00
|
|
|
td := setupHTMLFilesRecursively(t, DebugMode, false)
|
2018-03-23 01:51:42 +03:00
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoadHTMLFilesRecursively3(t *testing.T) {
|
|
|
|
td := setupHTMLFilesRecursively(t, ReleaseMode, false)
|
|
|
|
res, err := http.Get("http://127.0.0.1:8888/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestLoadHTMLFilesRecursivelyUsingTLS(t *testing.T) {
|
2018-03-23 21:14:34 +03:00
|
|
|
td := setupHTMLFilesRecursively(t, TestMode, true)
|
2018-03-23 01:51:42 +03:00
|
|
|
// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error
|
|
|
|
tr := &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
InsecureSkipVerify: true,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
client := &http.Client{Transport: tr}
|
|
|
|
res, err := client.Get("https://127.0.0.1:9999/test")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "<h1>Hello world</h1>", string(resp[:]))
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
2018-03-23 21:14:34 +03:00
|
|
|
func TestLoadHTMLFilesRecursivelyFuncMap(t *testing.T) {
|
2018-03-23 01:51:42 +03:00
|
|
|
time.Now()
|
2018-03-23 21:14:34 +03:00
|
|
|
td := setupHTMLFilesRecursively(t, TestMode, false)
|
2018-03-23 01:51:42 +03:00
|
|
|
res, err := http.Get("http://127.0.0.1:8888/raw")
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, _ := ioutil.ReadAll(res.Body)
|
|
|
|
assert.Equal(t, "Date: 2017/07/01\n", string(resp[:]))
|
|
|
|
|
|
|
|
td()
|
|
|
|
}
|
|
|
|
|
2015-06-04 05:32:18 +03:00
|
|
|
func TestAddRoute(t *testing.T) {
|
|
|
|
router := New()
|
|
|
|
router.addRoute("GET", "/", HandlersChain{func(_ *Context) {}})
|
|
|
|
|
|
|
|
assert.Len(t, router.trees, 1)
|
|
|
|
assert.NotNil(t, router.trees.get("GET"))
|
|
|
|
assert.Nil(t, router.trees.get("POST"))
|
|
|
|
|
|
|
|
router.addRoute("POST", "/", HandlersChain{func(_ *Context) {}})
|
2014-07-04 12:12:28 +04:00
|
|
|
|
2015-06-04 05:32:18 +03:00
|
|
|
assert.Len(t, router.trees, 2)
|
|
|
|
assert.NotNil(t, router.trees.get("GET"))
|
|
|
|
assert.NotNil(t, router.trees.get("POST"))
|
|
|
|
|
|
|
|
router.addRoute("POST", "/post", HandlersChain{func(_ *Context) {}})
|
|
|
|
assert.Len(t, router.trees, 2)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAddRouteFails(t *testing.T) {
|
|
|
|
router := New()
|
2015-05-20 00:22:35 +03:00
|
|
|
assert.Panics(t, func() { router.addRoute("", "/", HandlersChain{func(_ *Context) {}}) })
|
|
|
|
assert.Panics(t, func() { router.addRoute("GET", "a", HandlersChain{func(_ *Context) {}}) })
|
|
|
|
assert.Panics(t, func() { router.addRoute("GET", "/", HandlersChain{}) })
|
2015-06-04 05:32:18 +03:00
|
|
|
|
|
|
|
router.addRoute("POST", "/post", HandlersChain{func(_ *Context) {}})
|
|
|
|
assert.Panics(t, func() {
|
|
|
|
router.addRoute("POST", "/post", HandlersChain{func(_ *Context) {}})
|
|
|
|
})
|
2014-07-04 12:12:28 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestCreateDefaultRouter(t *testing.T) {
|
|
|
|
router := Default()
|
|
|
|
assert.Len(t, router.Handlers, 2)
|
2014-07-04 12:12:28 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestNoRouteWithoutGlobalHandlers(t *testing.T) {
|
2015-05-31 23:35:49 +03:00
|
|
|
var middleware0 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware1 HandlerFunc = func(c *Context) {}
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router := New()
|
|
|
|
|
|
|
|
router.NoRoute(middleware0)
|
|
|
|
assert.Nil(t, router.Handlers)
|
|
|
|
assert.Len(t, router.noRoute, 1)
|
|
|
|
assert.Len(t, router.allNoRoute, 1)
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.noRoute[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoRoute[0], middleware0)
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router.NoRoute(middleware1, middleware0)
|
|
|
|
assert.Len(t, router.noRoute, 2)
|
|
|
|
assert.Len(t, router.allNoRoute, 2)
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.noRoute[0], middleware1)
|
|
|
|
compareFunc(t, router.allNoRoute[0], middleware1)
|
|
|
|
compareFunc(t, router.noRoute[1], middleware0)
|
|
|
|
compareFunc(t, router.allNoRoute[1], middleware0)
|
2014-07-04 12:12:28 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestNoRouteWithGlobalHandlers(t *testing.T) {
|
2015-05-31 23:35:49 +03:00
|
|
|
var middleware0 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware1 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware2 HandlerFunc = func(c *Context) {}
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router := New()
|
|
|
|
router.Use(middleware2)
|
|
|
|
|
|
|
|
router.NoRoute(middleware0)
|
|
|
|
assert.Len(t, router.allNoRoute, 2)
|
|
|
|
assert.Len(t, router.Handlers, 1)
|
|
|
|
assert.Len(t, router.noRoute, 1)
|
|
|
|
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.Handlers[0], middleware2)
|
|
|
|
compareFunc(t, router.noRoute[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoRoute[0], middleware2)
|
|
|
|
compareFunc(t, router.allNoRoute[1], middleware0)
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router.Use(middleware1)
|
|
|
|
assert.Len(t, router.allNoRoute, 3)
|
|
|
|
assert.Len(t, router.Handlers, 2)
|
|
|
|
assert.Len(t, router.noRoute, 1)
|
|
|
|
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.Handlers[0], middleware2)
|
|
|
|
compareFunc(t, router.Handlers[1], middleware1)
|
|
|
|
compareFunc(t, router.noRoute[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoRoute[0], middleware2)
|
|
|
|
compareFunc(t, router.allNoRoute[1], middleware1)
|
|
|
|
compareFunc(t, router.allNoRoute[2], middleware0)
|
2014-07-04 12:12:28 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestNoMethodWithoutGlobalHandlers(t *testing.T) {
|
2015-05-31 23:35:49 +03:00
|
|
|
var middleware0 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware1 HandlerFunc = func(c *Context) {}
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router := New()
|
|
|
|
|
|
|
|
router.NoMethod(middleware0)
|
|
|
|
assert.Empty(t, router.Handlers)
|
|
|
|
assert.Len(t, router.noMethod, 1)
|
|
|
|
assert.Len(t, router.allNoMethod, 1)
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.noMethod[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoMethod[0], middleware0)
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router.NoMethod(middleware1, middleware0)
|
|
|
|
assert.Len(t, router.noMethod, 2)
|
|
|
|
assert.Len(t, router.allNoMethod, 2)
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.noMethod[0], middleware1)
|
|
|
|
compareFunc(t, router.allNoMethod[0], middleware1)
|
|
|
|
compareFunc(t, router.noMethod[1], middleware0)
|
|
|
|
compareFunc(t, router.allNoMethod[1], middleware0)
|
2014-07-04 12:12:28 +04:00
|
|
|
}
|
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestRebuild404Handlers(t *testing.T) {
|
2014-07-05 19:04:11 +04:00
|
|
|
|
|
|
|
}
|
2014-07-28 14:05:23 +04:00
|
|
|
|
2015-04-08 03:58:35 +03:00
|
|
|
func TestNoMethodWithGlobalHandlers(t *testing.T) {
|
2015-05-31 23:35:49 +03:00
|
|
|
var middleware0 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware1 HandlerFunc = func(c *Context) {}
|
|
|
|
var middleware2 HandlerFunc = func(c *Context) {}
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router := New()
|
|
|
|
router.Use(middleware2)
|
|
|
|
|
|
|
|
router.NoMethod(middleware0)
|
|
|
|
assert.Len(t, router.allNoMethod, 2)
|
|
|
|
assert.Len(t, router.Handlers, 1)
|
|
|
|
assert.Len(t, router.noMethod, 1)
|
|
|
|
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.Handlers[0], middleware2)
|
|
|
|
compareFunc(t, router.noMethod[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoMethod[0], middleware2)
|
|
|
|
compareFunc(t, router.allNoMethod[1], middleware0)
|
2015-04-08 03:58:35 +03:00
|
|
|
|
|
|
|
router.Use(middleware1)
|
|
|
|
assert.Len(t, router.allNoMethod, 3)
|
|
|
|
assert.Len(t, router.Handlers, 2)
|
|
|
|
assert.Len(t, router.noMethod, 1)
|
|
|
|
|
2015-05-31 23:35:49 +03:00
|
|
|
compareFunc(t, router.Handlers[0], middleware2)
|
|
|
|
compareFunc(t, router.Handlers[1], middleware1)
|
|
|
|
compareFunc(t, router.noMethod[0], middleware0)
|
|
|
|
compareFunc(t, router.allNoMethod[0], middleware2)
|
|
|
|
compareFunc(t, router.allNoMethod[1], middleware1)
|
|
|
|
compareFunc(t, router.allNoMethod[2], middleware0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func compareFunc(t *testing.T, a, b interface{}) {
|
|
|
|
sf1 := reflect.ValueOf(a)
|
|
|
|
sf2 := reflect.ValueOf(b)
|
|
|
|
if sf1.Pointer() != sf2.Pointer() {
|
|
|
|
t.Error("different functions")
|
|
|
|
}
|
2014-07-28 14:05:23 +04:00
|
|
|
}
|
2015-06-07 05:20:39 +03:00
|
|
|
|
|
|
|
func TestListOfRoutes(t *testing.T) {
|
|
|
|
router := New()
|
2016-04-15 02:16:46 +03:00
|
|
|
router.GET("/favicon.ico", handlerTest1)
|
|
|
|
router.GET("/", handlerTest1)
|
2015-06-07 05:20:39 +03:00
|
|
|
group := router.Group("/users")
|
|
|
|
{
|
2016-04-15 02:16:46 +03:00
|
|
|
group.GET("/", handlerTest2)
|
|
|
|
group.GET("/:id", handlerTest1)
|
|
|
|
group.POST("/:id", handlerTest2)
|
2015-06-07 05:20:39 +03:00
|
|
|
}
|
|
|
|
router.Static("/static", ".")
|
|
|
|
|
|
|
|
list := router.Routes()
|
|
|
|
|
|
|
|
assert.Len(t, list, 7)
|
2015-08-21 21:57:53 +03:00
|
|
|
assertRoutePresent(t, list, RouteInfo{
|
2015-06-18 18:17:22 +03:00
|
|
|
Method: "GET",
|
|
|
|
Path: "/favicon.ico",
|
2016-04-15 02:35:22 +03:00
|
|
|
Handler: "^(.*/vendor/)?github.com/gin-gonic/gin.handlerTest1$",
|
2015-06-07 05:20:39 +03:00
|
|
|
})
|
2015-08-21 21:57:53 +03:00
|
|
|
assertRoutePresent(t, list, RouteInfo{
|
2015-06-18 18:17:22 +03:00
|
|
|
Method: "GET",
|
|
|
|
Path: "/",
|
2016-04-15 02:35:22 +03:00
|
|
|
Handler: "^(.*/vendor/)?github.com/gin-gonic/gin.handlerTest1$",
|
2015-06-07 05:20:39 +03:00
|
|
|
})
|
2015-08-21 21:57:53 +03:00
|
|
|
assertRoutePresent(t, list, RouteInfo{
|
2015-06-18 18:17:22 +03:00
|
|
|
Method: "GET",
|
|
|
|
Path: "/users/",
|
2016-04-15 02:35:22 +03:00
|
|
|
Handler: "^(.*/vendor/)?github.com/gin-gonic/gin.handlerTest2$",
|
2015-06-07 05:20:39 +03:00
|
|
|
})
|
2015-08-21 21:57:53 +03:00
|
|
|
assertRoutePresent(t, list, RouteInfo{
|
2015-06-18 18:17:22 +03:00
|
|
|
Method: "GET",
|
|
|
|
Path: "/users/:id",
|
2016-04-15 02:35:22 +03:00
|
|
|
Handler: "^(.*/vendor/)?github.com/gin-gonic/gin.handlerTest1$",
|
2015-06-07 05:20:39 +03:00
|
|
|
})
|
2015-08-21 21:57:53 +03:00
|
|
|
assertRoutePresent(t, list, RouteInfo{
|
2015-06-18 18:17:22 +03:00
|
|
|
Method: "POST",
|
|
|
|
Path: "/users/:id",
|
2016-04-15 02:35:22 +03:00
|
|
|
Handler: "^(.*/vendor/)?github.com/gin-gonic/gin.handlerTest2$",
|
2015-06-07 05:20:39 +03:00
|
|
|
})
|
|
|
|
}
|
2015-06-18 18:17:22 +03:00
|
|
|
|
2019-01-18 04:57:06 +03:00
|
|
|
func TestEngineHandleContext(t *testing.T) {
|
|
|
|
r := New()
|
|
|
|
r.GET("/", func(c *Context) {
|
|
|
|
c.Request.URL.Path = "/v2"
|
|
|
|
r.HandleContext(c)
|
|
|
|
})
|
|
|
|
v2 := r.Group("/v2")
|
|
|
|
{
|
|
|
|
v2.GET("/", func(c *Context) {})
|
|
|
|
}
|
|
|
|
|
|
|
|
assert.NotPanics(t, func() {
|
|
|
|
w := performRequest(r, "GET", "/")
|
|
|
|
assert.Equal(t, 301, w.Code)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-02-18 04:35:08 +03:00
|
|
|
func TestEngineHandleContextManyReEntries(t *testing.T) {
|
|
|
|
expectValue := 10000
|
|
|
|
|
|
|
|
var handlerCounter, middlewareCounter int64
|
|
|
|
|
|
|
|
r := New()
|
|
|
|
r.Use(func(c *Context) {
|
|
|
|
atomic.AddInt64(&middlewareCounter, 1)
|
|
|
|
})
|
|
|
|
r.GET("/:count", func(c *Context) {
|
|
|
|
countStr := c.Param("count")
|
|
|
|
count, err := strconv.Atoi(countStr)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
|
|
|
|
n, err := c.Writer.Write([]byte("."))
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, 1, n)
|
|
|
|
|
|
|
|
switch {
|
|
|
|
case count > 0:
|
|
|
|
c.Request.URL.Path = "/" + strconv.Itoa(count-1)
|
|
|
|
r.HandleContext(c)
|
|
|
|
}
|
|
|
|
}, func(c *Context) {
|
|
|
|
atomic.AddInt64(&handlerCounter, 1)
|
|
|
|
})
|
|
|
|
|
|
|
|
assert.NotPanics(t, func() {
|
|
|
|
w := performRequest(r, "GET", "/"+strconv.Itoa(expectValue-1)) // include 0 value
|
|
|
|
assert.Equal(t, 200, w.Code)
|
|
|
|
assert.Equal(t, expectValue, w.Body.Len())
|
|
|
|
})
|
|
|
|
|
|
|
|
assert.Equal(t, int64(expectValue), handlerCounter)
|
|
|
|
assert.Equal(t, int64(expectValue), middlewareCounter)
|
|
|
|
}
|
|
|
|
|
2021-04-06 06:37:25 +03:00
|
|
|
func TestPrepareTrustedCIRDsWith(t *testing.T) {
|
|
|
|
r := New()
|
|
|
|
|
|
|
|
// valid ipv4 cidr
|
|
|
|
{
|
|
|
|
expectedTrustedCIDRs := []*net.IPNet{parseCIDR("0.0.0.0/0")}
|
|
|
|
r.TrustedProxies = []string{"0.0.0.0/0"}
|
|
|
|
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedTrustedCIDRs, trustedCIDRs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid ipv4 cidr
|
|
|
|
{
|
|
|
|
r.TrustedProxies = []string{"192.168.1.33/33"}
|
|
|
|
|
|
|
|
_, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid ipv4 address
|
|
|
|
{
|
|
|
|
expectedTrustedCIDRs := []*net.IPNet{parseCIDR("192.168.1.33/32")}
|
|
|
|
r.TrustedProxies = []string{"192.168.1.33"}
|
|
|
|
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedTrustedCIDRs, trustedCIDRs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid ipv4 address
|
|
|
|
{
|
|
|
|
r.TrustedProxies = []string{"192.168.1.256"}
|
|
|
|
|
|
|
|
_, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid ipv6 address
|
|
|
|
{
|
|
|
|
expectedTrustedCIDRs := []*net.IPNet{parseCIDR("2002:0000:0000:1234:abcd:ffff:c0a8:0101/128")}
|
|
|
|
r.TrustedProxies = []string{"2002:0000:0000:1234:abcd:ffff:c0a8:0101"}
|
|
|
|
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedTrustedCIDRs, trustedCIDRs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid ipv6 address
|
|
|
|
{
|
|
|
|
r.TrustedProxies = []string{"gggg:0000:0000:1234:abcd:ffff:c0a8:0101"}
|
|
|
|
|
|
|
|
_, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid ipv6 cidr
|
|
|
|
{
|
|
|
|
expectedTrustedCIDRs := []*net.IPNet{parseCIDR("::/0")}
|
|
|
|
r.TrustedProxies = []string{"::/0"}
|
|
|
|
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedTrustedCIDRs, trustedCIDRs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid ipv6 cidr
|
|
|
|
{
|
|
|
|
r.TrustedProxies = []string{"gggg:0000:0000:1234:abcd:ffff:c0a8:0101/129"}
|
|
|
|
|
|
|
|
_, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// valid combination
|
|
|
|
{
|
|
|
|
expectedTrustedCIDRs := []*net.IPNet{
|
|
|
|
parseCIDR("::/0"),
|
|
|
|
parseCIDR("192.168.0.0/16"),
|
|
|
|
parseCIDR("172.16.0.1/32"),
|
|
|
|
}
|
|
|
|
r.TrustedProxies = []string{
|
|
|
|
"::/0",
|
|
|
|
"192.168.0.0/16",
|
|
|
|
"172.16.0.1",
|
|
|
|
}
|
|
|
|
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.NoError(t, err)
|
|
|
|
assert.Equal(t, expectedTrustedCIDRs, trustedCIDRs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// invalid combination
|
|
|
|
{
|
|
|
|
r.TrustedProxies = []string{
|
|
|
|
"::/0",
|
|
|
|
"192.168.0.0/16",
|
|
|
|
"172.16.0.256",
|
|
|
|
}
|
|
|
|
_, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// nil value
|
|
|
|
{
|
|
|
|
r.TrustedProxies = nil
|
|
|
|
trustedCIDRs, err := r.prepareTrustedCIDRs()
|
|
|
|
|
|
|
|
assert.Nil(t, trustedCIDRs)
|
|
|
|
assert.Nil(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseCIDR(cidr string) *net.IPNet {
|
|
|
|
_, parsedCIDR, err := net.ParseCIDR(cidr)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
}
|
|
|
|
return parsedCIDR
|
|
|
|
}
|
|
|
|
|
2015-08-21 21:57:53 +03:00
|
|
|
func assertRoutePresent(t *testing.T, gotRoutes RoutesInfo, wantRoute RouteInfo) {
|
|
|
|
for _, gotRoute := range gotRoutes {
|
|
|
|
if gotRoute.Path == wantRoute.Path && gotRoute.Method == wantRoute.Method {
|
2016-03-09 10:29:01 +03:00
|
|
|
assert.Regexp(t, wantRoute.Handler, gotRoute.Handler)
|
2015-08-21 21:57:53 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
t.Errorf("route not found: %v", wantRoute)
|
|
|
|
}
|
|
|
|
|
2016-04-15 02:16:46 +03:00
|
|
|
func handlerTest1(c *Context) {}
|
|
|
|
func handlerTest2(c *Context) {}
|