mirror of
				https://github.com/gin-gonic/gin.git
				synced 2025-10-22 01:12:16 +08:00 
			
		
		
		
	Merge branch 'master' into master
This commit is contained in:
		
						commit
						efe7384a24
					
				
							
								
								
									
										13
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								.github/ISSUE_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,13 @@ | |||||||
|  | - With issues: | ||||||
|  |   - Use the search tool before opening a new issue. | ||||||
|  |   - Please provide source code and commit sha if you found a bug. | ||||||
|  |   - Review existing issues and provide feedback or react to them. | ||||||
|  | 
 | ||||||
|  | - go version: | ||||||
|  | - gin version (or commit ref): | ||||||
|  | - operating system: | ||||||
|  | 
 | ||||||
|  | ## Description | ||||||
|  | 
 | ||||||
|  | ## Screenshots | ||||||
|  | 
 | ||||||
							
								
								
									
										7
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								.github/PULL_REQUEST_TEMPLATE.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | - With pull requests: | ||||||
|  |   - Open your pull request against `master` | ||||||
|  |   - Your pull request should have no more than two commits, if not you should squash them. | ||||||
|  |   - It should pass all tests in the available continuous integrations systems such as TravisCI. | ||||||
|  |   - You should add/modify tests to cover your proposed code changes. | ||||||
|  |   - If your pull request contains a new feature, please document it on the README. | ||||||
|  | 
 | ||||||
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -3,3 +3,5 @@ vendor/* | |||||||
| coverage.out | coverage.out | ||||||
| count.out | count.out | ||||||
| test | test | ||||||
|  | profile.out | ||||||
|  | tmp.out | ||||||
|  | |||||||
							
								
								
									
										17
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										17
									
								
								.travis.yml
									
									
									
									
									
								
							| @ -5,13 +5,26 @@ go: | |||||||
|   - 1.7.x |   - 1.7.x | ||||||
|   - 1.8.x |   - 1.8.x | ||||||
|   - 1.9.x |   - 1.9.x | ||||||
|  |   - 1.10.x | ||||||
|  |   - 1.11.x | ||||||
|   - master |   - master | ||||||
| 
 | 
 | ||||||
|  | matrix: | ||||||
|  |   fast_finish: true | ||||||
|  |   include: | ||||||
|  |   - go: 1.11.x | ||||||
|  |     env: GO111MODULE=on | ||||||
|  | 
 | ||||||
| git: | git: | ||||||
|   depth: 3 |   depth: 10 | ||||||
|  | 
 | ||||||
|  | before_install: | ||||||
|  |   - if [[ "${GO111MODULE}" = "on" ]]; then mkdir "${HOME}/go"; export GOPATH="${HOME}/go"; fi | ||||||
| 
 | 
 | ||||||
| install: | install: | ||||||
|   - make install |   - if [[ "${GO111MODULE}" = "on" ]]; then go mod download; else make install; fi | ||||||
|  |   - if [[ "${GO111MODULE}" = "on" ]]; then export PATH="${GOPATH}/bin:${GOROOT}/bin:${PATH}"; fi | ||||||
|  |   - if [[ "${GO111MODULE}" = "on" ]]; then make tools; fi | ||||||
| 
 | 
 | ||||||
| go_import_path: github.com/gin-gonic/gin | go_import_path: github.com/gin-gonic/gin | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,8 +1,12 @@ | |||||||
| List of all the awesome people working to make Gin the best Web Framework in Go. | List of all the awesome people working to make Gin the best Web Framework in Go. | ||||||
| 
 | 
 | ||||||
|  | ## gin 1.x series authors | ||||||
|  | 
 | ||||||
|  | **Gin Core Team:** Bo-Yi Wu (@appleboy), 田欧 (@thinkerou), Javier Provecho (@javierprovecho) | ||||||
|  | 
 | ||||||
| ## gin 0.x series authors | ## gin 0.x series authors | ||||||
| 
 | 
 | ||||||
| **Maintainer:** Manu Martinez-Almeida (@manucorporat), Javier Provecho (@javierprovecho) | **Maintainers:** Manu Martinez-Almeida (@manucorporat), Javier Provecho (@javierprovecho) | ||||||
| 
 | 
 | ||||||
| People and companies, who have contributed, in alphabetical order. | People and companies, who have contributed, in alphabetical order. | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										24
									
								
								CHANGELOG.md
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								CHANGELOG.md
									
									
									
									
									
								
							| @ -1,6 +1,28 @@ | |||||||
| # CHANGELOG | # CHANGELOG | ||||||
| 
 | 
 | ||||||
| ### Gin 1.2 | ### Gin 1.3.0 | ||||||
|  | 
 | ||||||
|  | - [NEW] Add [`func (*Context) QueryMap`](https://godoc.org/github.com/gin-gonic/gin#Context.QueryMap), [`func (*Context) GetQueryMap`](https://godoc.org/github.com/gin-gonic/gin#Context.GetQueryMap), [`func (*Context) PostFormMap`](https://godoc.org/github.com/gin-gonic/gin#Context.PostFormMap) and [`func (*Context) GetPostFormMap`](https://godoc.org/github.com/gin-gonic/gin#Context.GetPostFormMap) to support `type map[string]string` as query string or form parameters, see [#1383](https://github.com/gin-gonic/gin/pull/1383) | ||||||
|  | - [NEW] Add [`func (*Context) AsciiJSON`](https://godoc.org/github.com/gin-gonic/gin#Context.AsciiJSON), see [#1358](https://github.com/gin-gonic/gin/pull/1358) | ||||||
|  | - [NEW] Add `Pusher()` in [`type ResponseWriter`](https://godoc.org/github.com/gin-gonic/gin#ResponseWriter) for supporting http2 push, see [#1273](https://github.com/gin-gonic/gin/pull/1273) | ||||||
|  | - [NEW] Add [`func (*Context) DataFromReader`](https://godoc.org/github.com/gin-gonic/gin#Context.DataFromReader) for serving dynamic data, see [#1304](https://github.com/gin-gonic/gin/pull/1304) | ||||||
|  | - [NEW] Add [`func (*Context) ShouldBindBodyWith`](https://godoc.org/github.com/gin-gonic/gin#Context.ShouldBindBodyWith) allowing to call binding multiple times, see [#1341](https://github.com/gin-gonic/gin/pull/1341) | ||||||
|  | - [NEW] Support pointers in form binding, see [#1336](https://github.com/gin-gonic/gin/pull/1336) | ||||||
|  | - [NEW] Add [`func (*Context) JSONP`](https://godoc.org/github.com/gin-gonic/gin#Context.JSONP), see [#1333](https://github.com/gin-gonic/gin/pull/1333) | ||||||
|  | - [NEW] Support default value in form binding, see [#1138](https://github.com/gin-gonic/gin/pull/1138) | ||||||
|  | - [NEW] Expose validator engine in [`type StructValidator`](https://godoc.org/github.com/gin-gonic/gin/binding#StructValidator), see [#1277](https://github.com/gin-gonic/gin/pull/1277) | ||||||
|  | - [NEW] Add [`func (*Context) ShouldBind`](https://godoc.org/github.com/gin-gonic/gin#Context.ShouldBind), [`func (*Context) ShouldBindQuery`](https://godoc.org/github.com/gin-gonic/gin#Context.ShouldBindQuery) and [`func (*Context) ShouldBindJSON`](https://godoc.org/github.com/gin-gonic/gin#Context.ShouldBindJSON), see [#1047](https://github.com/gin-gonic/gin/pull/1047) | ||||||
|  | - [NEW] Add support for `time.Time` location in form binding, see [#1117](https://github.com/gin-gonic/gin/pull/1117) | ||||||
|  | - [NEW] Add [`func (*Context) BindQuery`](https://godoc.org/github.com/gin-gonic/gin#Context.BindQuery), see [#1029](https://github.com/gin-gonic/gin/pull/1029) | ||||||
|  | - [NEW] Make [jsonite](https://github.com/json-iterator/go) optional with build tags, see [#1026](https://github.com/gin-gonic/gin/pull/1026) | ||||||
|  | - [NEW] Show query string in logger, see [#999](https://github.com/gin-gonic/gin/pull/999) | ||||||
|  | - [NEW] Add [`func (*Context) SecureJSON`](https://godoc.org/github.com/gin-gonic/gin#Context.SecureJSON), see [#987](https://github.com/gin-gonic/gin/pull/987) and [#993](https://github.com/gin-gonic/gin/pull/993) | ||||||
|  | - [DEPRECATE] `func (*Context) GetCookie` for [`func (*Context) Cookie`](https://godoc.org/github.com/gin-gonic/gin#Context.Cookie) | ||||||
|  | - [FIX] Don't display color tags if [`func DisableConsoleColor`](https://godoc.org/github.com/gin-gonic/gin#DisableConsoleColor) called, see [#1072](https://github.com/gin-gonic/gin/pull/1072) | ||||||
|  | - [FIX] Gin Mode `""` when calling [`func Mode`](https://godoc.org/github.com/gin-gonic/gin#Mode) now returns `const DebugMode`, see [#1250](https://github.com/gin-gonic/gin/pull/1250) | ||||||
|  | - [FIX] `Flush()` now doesn't overwrite `responseWriter` status code, see [#1460](https://github.com/gin-gonic/gin/pull/1460) | ||||||
|  | 
 | ||||||
|  | ### Gin 1.2.0 | ||||||
| 
 | 
 | ||||||
| - [NEW] Switch from godeps to govendor | - [NEW] Switch from godeps to govendor | ||||||
| - [NEW] Add support for Let's Encrypt via gin-gonic/autotls | - [NEW] Add support for Let's Encrypt via gin-gonic/autotls | ||||||
|  | |||||||
							
								
								
									
										44
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										44
									
								
								Makefile
									
									
									
									
									
								
							| @ -1,6 +1,9 @@ | |||||||
|  | GO ?= go | ||||||
| GOFMT ?= gofmt "-s" | GOFMT ?= gofmt "-s" | ||||||
| PACKAGES ?= $(shell go list ./... | grep -v /vendor/) | PACKAGES ?= $(shell $(GO) list ./... | grep -v /vendor/) | ||||||
|  | VETPACKAGES ?= $(shell $(GO) list ./... | grep -v /vendor/ | grep -v /examples/) | ||||||
| GOFILES := $(shell find . -name "*.go" -type f -not -path "./vendor/*") | GOFILES := $(shell find . -name "*.go" -type f -not -path "./vendor/*") | ||||||
|  | TESTFOLDER := $(shell $(GO) list ./... | grep -E 'gin$$|binding$$|render$$' | grep -v examples) | ||||||
| 
 | 
 | ||||||
| all: install | all: install | ||||||
| 
 | 
 | ||||||
| @ -9,7 +12,25 @@ install: deps | |||||||
| 
 | 
 | ||||||
| .PHONY: test | .PHONY: test | ||||||
| test: | test: | ||||||
| 	sh coverage.sh | 	echo "mode: count" > coverage.out | ||||||
|  | 	for d in $(TESTFOLDER); do \
 | ||||||
|  | 		$(GO) test -v -covermode=count -coverprofile=profile.out $$d > tmp.out; \
 | ||||||
|  | 		cat tmp.out; \
 | ||||||
|  | 		if grep -q "^--- FAIL" tmp.out; then \
 | ||||||
|  | 			rm tmp.out; \
 | ||||||
|  | 			exit 1; \
 | ||||||
|  | 		elif grep -q "build failed" tmp.out; then \
 | ||||||
|  | 			rm tmp.out; \
 | ||||||
|  | 			exit 1; \
 | ||||||
|  | 		elif grep -q "setup failed" tmp.out; then \
 | ||||||
|  | 			rm tmp.out; \
 | ||||||
|  | 			exit 1; \
 | ||||||
|  | 		fi; \
 | ||||||
|  | 		if [ -f profile.out ]; then \
 | ||||||
|  | 			cat profile.out | grep -v "mode:" >> coverage.out; \
 | ||||||
|  | 			rm profile.out; \
 | ||||||
|  | 		fi; \
 | ||||||
|  | 	done | ||||||
| 
 | 
 | ||||||
| .PHONY: fmt | .PHONY: fmt | ||||||
| fmt: | fmt: | ||||||
| @ -17,7 +38,6 @@ fmt: | |||||||
| 
 | 
 | ||||||
| .PHONY: fmt-check | .PHONY: fmt-check | ||||||
| fmt-check: | fmt-check: | ||||||
| 	# get all go files and run go fmt on them |  | ||||||
| 	@diff=$$($(GOFMT) -d $(GOFILES)); \
 | 	@diff=$$($(GOFMT) -d $(GOFILES)); \
 | ||||||
| 	if [ -n "$$diff" ]; then \
 | 	if [ -n "$$diff" ]; then \
 | ||||||
| 		echo "Please run 'make fmt' and commit the result:"; \
 | 		echo "Please run 'make fmt' and commit the result:"; \
 | ||||||
| @ -26,14 +46,14 @@ fmt-check: | |||||||
| 	fi; | 	fi; | ||||||
| 
 | 
 | ||||||
| vet: | vet: | ||||||
| 	go vet $(PACKAGES) | 	$(GO) vet $(VETPACKAGES) | ||||||
| 
 | 
 | ||||||
| deps: | deps: | ||||||
| 	@hash govendor > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 	@hash govendor > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | ||||||
| 		go get -u github.com/kardianos/govendor; \
 | 		$(GO) get -u github.com/kardianos/govendor; \
 | ||||||
| 	fi | 	fi | ||||||
| 	@hash embedmd > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 	@hash embedmd > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | ||||||
| 		go get -u github.com/campoy/embedmd; \
 | 		$(GO) get -u github.com/campoy/embedmd; \
 | ||||||
| 	fi | 	fi | ||||||
| 
 | 
 | ||||||
| embedmd: | embedmd: | ||||||
| @ -42,20 +62,26 @@ embedmd: | |||||||
| .PHONY: lint | .PHONY: lint | ||||||
| lint: | lint: | ||||||
| 	@hash golint > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 	@hash golint > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | ||||||
| 		go get -u github.com/golang/lint/golint; \
 | 		$(GO) get -u golang.org/x/lint/golint; \
 | ||||||
| 	fi | 	fi | ||||||
| 	for PKG in $(PACKAGES); do golint -set_exit_status $$PKG || exit 1; done; | 	for PKG in $(PACKAGES); do golint -set_exit_status $$PKG || exit 1; done; | ||||||
| 
 | 
 | ||||||
| .PHONY: misspell-check | .PHONY: misspell-check | ||||||
| misspell-check: | misspell-check: | ||||||
| 	@hash misspell > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 	@hash misspell > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | ||||||
| 		go get -u github.com/client9/misspell/cmd/misspell; \
 | 		$(GO) get -u github.com/client9/misspell/cmd/misspell; \
 | ||||||
| 	fi | 	fi | ||||||
| 	misspell -error $(GOFILES) | 	misspell -error $(GOFILES) | ||||||
| 
 | 
 | ||||||
| .PHONY: misspell | .PHONY: misspell | ||||||
| misspell: | misspell: | ||||||
| 	@hash misspell > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | 	@hash misspell > /dev/null 2>&1; if [ $$? -ne 0 ]; then \
 | ||||||
| 		go get -u github.com/client9/misspell/cmd/misspell; \
 | 		$(GO) get -u github.com/client9/misspell/cmd/misspell; \
 | ||||||
| 	fi | 	fi | ||||||
| 	misspell -w $(GOFILES) | 	misspell -w $(GOFILES) | ||||||
|  | 
 | ||||||
|  | .PHONY: tools | ||||||
|  | tools: | ||||||
|  | 	go install golang.org/x/lint/golint; \
 | ||||||
|  | 	go install github.com/client9/misspell/cmd/misspell; \
 | ||||||
|  | 	go install github.com/campoy/embedmd; | ||||||
|  | |||||||
							
								
								
									
										3
									
								
								auth.go
									
									
									
									
									
								
							
							
						
						
									
										3
									
								
								auth.go
									
									
									
									
									
								
							| @ -7,6 +7,7 @@ package gin | |||||||
| import ( | import ( | ||||||
| 	"crypto/subtle" | 	"crypto/subtle" | ||||||
| 	"encoding/base64" | 	"encoding/base64" | ||||||
|  | 	"net/http" | ||||||
| 	"strconv" | 	"strconv" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -51,7 +52,7 @@ func BasicAuthForRealm(accounts Accounts, realm string) HandlerFunc { | |||||||
| 		if !found { | 		if !found { | ||||||
| 			// Credentials doesn't match, we return 401 and abort handlers chain. | 			// Credentials doesn't match, we return 401 and abort handlers chain. | ||||||
| 			c.Header("WWW-Authenticate", realm) | 			c.Header("WWW-Authenticate", realm) | ||||||
| 			c.AbortWithStatus(401) | 			c.AbortWithStatus(http.StatusUnauthorized) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										16
									
								
								auth_test.go
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								auth_test.go
									
									
									
									
									
								
							| @ -93,7 +93,7 @@ func TestBasicAuthSucceed(t *testing.T) { | |||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(BasicAuth(accounts)) | 	router.Use(BasicAuth(accounts)) | ||||||
| 	router.GET("/login", func(c *Context) { | 	router.GET("/login", func(c *Context) { | ||||||
| 		c.String(200, c.MustGet(AuthUserKey).(string)) | 		c.String(http.StatusOK, c.MustGet(AuthUserKey).(string)) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	w := httptest.NewRecorder() | 	w := httptest.NewRecorder() | ||||||
| @ -101,7 +101,7 @@ func TestBasicAuthSucceed(t *testing.T) { | |||||||
| 	req.Header.Set("Authorization", authorizationHeader("admin", "password")) | 	req.Header.Set("Authorization", authorizationHeader("admin", "password")) | ||||||
| 	router.ServeHTTP(w, req) | 	router.ServeHTTP(w, req) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, 200, w.Code) | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
| 	assert.Equal(t, "admin", w.Body.String()) | 	assert.Equal(t, "admin", w.Body.String()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -112,7 +112,7 @@ func TestBasicAuth401(t *testing.T) { | |||||||
| 	router.Use(BasicAuth(accounts)) | 	router.Use(BasicAuth(accounts)) | ||||||
| 	router.GET("/login", func(c *Context) { | 	router.GET("/login", func(c *Context) { | ||||||
| 		called = true | 		called = true | ||||||
| 		c.String(200, c.MustGet(AuthUserKey).(string)) | 		c.String(http.StatusOK, c.MustGet(AuthUserKey).(string)) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	w := httptest.NewRecorder() | 	w := httptest.NewRecorder() | ||||||
| @ -121,8 +121,8 @@ func TestBasicAuth401(t *testing.T) { | |||||||
| 	router.ServeHTTP(w, req) | 	router.ServeHTTP(w, req) | ||||||
| 
 | 
 | ||||||
| 	assert.False(t, called) | 	assert.False(t, called) | ||||||
| 	assert.Equal(t, 401, w.Code) | 	assert.Equal(t, http.StatusUnauthorized, w.Code) | ||||||
| 	assert.Equal(t, "Basic realm=\"Authorization Required\"", w.HeaderMap.Get("WWW-Authenticate")) | 	assert.Equal(t, "Basic realm=\"Authorization Required\"", w.Header().Get("WWW-Authenticate")) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBasicAuth401WithCustomRealm(t *testing.T) { | func TestBasicAuth401WithCustomRealm(t *testing.T) { | ||||||
| @ -132,7 +132,7 @@ func TestBasicAuth401WithCustomRealm(t *testing.T) { | |||||||
| 	router.Use(BasicAuthForRealm(accounts, "My Custom \"Realm\"")) | 	router.Use(BasicAuthForRealm(accounts, "My Custom \"Realm\"")) | ||||||
| 	router.GET("/login", func(c *Context) { | 	router.GET("/login", func(c *Context) { | ||||||
| 		called = true | 		called = true | ||||||
| 		c.String(200, c.MustGet(AuthUserKey).(string)) | 		c.String(http.StatusOK, c.MustGet(AuthUserKey).(string)) | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	w := httptest.NewRecorder() | 	w := httptest.NewRecorder() | ||||||
| @ -141,6 +141,6 @@ func TestBasicAuth401WithCustomRealm(t *testing.T) { | |||||||
| 	router.ServeHTTP(w, req) | 	router.ServeHTTP(w, req) | ||||||
| 
 | 
 | ||||||
| 	assert.False(t, called) | 	assert.False(t, called) | ||||||
| 	assert.Equal(t, 401, w.Code) | 	assert.Equal(t, http.StatusUnauthorized, w.Code) | ||||||
| 	assert.Equal(t, "Basic realm=\"My Custom \\\"Realm\\\"\"", w.HeaderMap.Get("WWW-Authenticate")) | 	assert.Equal(t, "Basic realm=\"My Custom \\\"Realm\\\"\"", w.Header().Get("WWW-Authenticate")) | ||||||
| } | } | ||||||
|  | |||||||
| @ -54,7 +54,7 @@ func BenchmarkOneRouteJSON(B *testing.B) { | |||||||
| 		Status string `json:"status"` | 		Status string `json:"status"` | ||||||
| 	}{"ok"} | 	}{"ok"} | ||||||
| 	router.GET("/json", func(c *Context) { | 	router.GET("/json", func(c *Context) { | ||||||
| 		c.JSON(200, data) | 		c.JSON(http.StatusOK, data) | ||||||
| 	}) | 	}) | ||||||
| 	runRequest(B, router, "GET", "/json") | 	runRequest(B, router, "GET", "/json") | ||||||
| } | } | ||||||
| @ -66,7 +66,7 @@ func BenchmarkOneRouteHTML(B *testing.B) { | |||||||
| 	router.SetHTMLTemplate(t) | 	router.SetHTMLTemplate(t) | ||||||
| 
 | 
 | ||||||
| 	router.GET("/html", func(c *Context) { | 	router.GET("/html", func(c *Context) { | ||||||
| 		c.HTML(200, "index", "hola") | 		c.HTML(http.StatusOK, "index", "hola") | ||||||
| 	}) | 	}) | ||||||
| 	runRequest(B, router, "GET", "/html") | 	runRequest(B, router, "GET", "/html") | ||||||
| } | } | ||||||
| @ -82,7 +82,7 @@ func BenchmarkOneRouteSet(B *testing.B) { | |||||||
| func BenchmarkOneRouteString(B *testing.B) { | func BenchmarkOneRouteString(B *testing.B) { | ||||||
| 	router := New() | 	router := New() | ||||||
| 	router.GET("/text", func(c *Context) { | 	router.GET("/text", func(c *Context) { | ||||||
| 		c.String(200, "this is a plain text") | 		c.String(http.StatusOK, "this is a plain text") | ||||||
| 	}) | 	}) | ||||||
| 	runRequest(B, router, "GET", "/text") | 	runRequest(B, router, "GET", "/text") | ||||||
| } | } | ||||||
|  | |||||||
| @ -4,12 +4,9 @@ | |||||||
| 
 | 
 | ||||||
| package binding | package binding | ||||||
| 
 | 
 | ||||||
| import ( | import "net/http" | ||||||
| 	"net/http" |  | ||||||
| 
 |  | ||||||
| 	"gopkg.in/go-playground/validator.v8" |  | ||||||
| ) |  | ||||||
| 
 | 
 | ||||||
|  | // Content-Type MIME of the most common data formats. | ||||||
| const ( | const ( | ||||||
| 	MIMEJSON              = "application/json" | 	MIMEJSON              = "application/json" | ||||||
| 	MIMEHTML              = "text/html" | 	MIMEHTML              = "text/html" | ||||||
| @ -21,13 +18,35 @@ const ( | |||||||
| 	MIMEPROTOBUF          = "application/x-protobuf" | 	MIMEPROTOBUF          = "application/x-protobuf" | ||||||
| 	MIMEMSGPACK           = "application/x-msgpack" | 	MIMEMSGPACK           = "application/x-msgpack" | ||||||
| 	MIMEMSGPACK2          = "application/msgpack" | 	MIMEMSGPACK2          = "application/msgpack" | ||||||
|  | 	MIMEYAML              = "application/x-yaml" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Binding describes the interface which needs to be implemented for binding the | ||||||
|  | // data present in the request such as JSON request body, query parameters or | ||||||
|  | // the form POST. | ||||||
| type Binding interface { | type Binding interface { | ||||||
| 	Name() string | 	Name() string | ||||||
| 	Bind(*http.Request, interface{}) error | 	Bind(*http.Request, interface{}) error | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // BindingBody adds BindBody method to Binding. BindBody is similar with Bind, | ||||||
|  | // but it reads the body from supplied bytes instead of req.Body. | ||||||
|  | type BindingBody interface { | ||||||
|  | 	Binding | ||||||
|  | 	BindBody([]byte, interface{}) error | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // BindingUri adds BindUri method to Binding. BindUri is similar with Bind, | ||||||
|  | // but it read the Params. | ||||||
|  | type BindingUri interface { | ||||||
|  | 	Name() string | ||||||
|  | 	BindUri(map[string][]string, interface{}) error | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // StructValidator is the minimal interface which needs to be implemented in | ||||||
|  | // order for it to be used as the validator engine for ensuring the correctness | ||||||
|  | // of the request. Gin provides a default implementation for this using | ||||||
|  | // https://github.com/go-playground/validator/tree/v8.18.2. | ||||||
| type StructValidator interface { | type StructValidator interface { | ||||||
| 	// ValidateStruct can receive any kind of type and it should never panic, even if the configuration is not right. | 	// ValidateStruct can receive any kind of type and it should never panic, even if the configuration is not right. | ||||||
| 	// If the received type is not a struct, any validation should be skipped and nil must be returned. | 	// If the received type is not a struct, any validation should be skipped and nil must be returned. | ||||||
| @ -36,14 +55,18 @@ type StructValidator interface { | |||||||
| 	// Otherwise nil must be returned. | 	// Otherwise nil must be returned. | ||||||
| 	ValidateStruct(interface{}) error | 	ValidateStruct(interface{}) error | ||||||
| 
 | 
 | ||||||
| 	// RegisterValidation adds a validation Func to a Validate's map of validators denoted by the key | 	// Engine returns the underlying validator engine which powers the | ||||||
| 	// NOTE: if the key already exists, the previous validation function will be replaced. | 	// StructValidator implementation. | ||||||
| 	// NOTE: this method is not thread-safe it is intended that these all be registered prior to any validation | 	Engine() interface{} | ||||||
| 	RegisterValidation(string, validator.Func) error |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Validator is the default validator which implements the StructValidator | ||||||
|  | // interface. It uses https://github.com/go-playground/validator/tree/v8.18.2 | ||||||
|  | // under the hood. | ||||||
| var Validator StructValidator = &defaultValidator{} | var Validator StructValidator = &defaultValidator{} | ||||||
| 
 | 
 | ||||||
|  | // These implement the Binding interface and can be used to bind the data | ||||||
|  | // present in the request to struct instances. | ||||||
| var ( | var ( | ||||||
| 	JSON          = jsonBinding{} | 	JSON          = jsonBinding{} | ||||||
| 	XML           = xmlBinding{} | 	XML           = xmlBinding{} | ||||||
| @ -53,8 +76,12 @@ var ( | |||||||
| 	FormMultipart = formMultipartBinding{} | 	FormMultipart = formMultipartBinding{} | ||||||
| 	ProtoBuf      = protobufBinding{} | 	ProtoBuf      = protobufBinding{} | ||||||
| 	MsgPack       = msgpackBinding{} | 	MsgPack       = msgpackBinding{} | ||||||
|  | 	YAML          = yamlBinding{} | ||||||
|  | 	Uri           = uriBinding{} | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Default returns the appropriate Binding instance based on the HTTP method | ||||||
|  | // and the content type. | ||||||
| func Default(method, contentType string) Binding { | func Default(method, contentType string) Binding { | ||||||
| 	if method == "GET" { | 	if method == "GET" { | ||||||
| 		return Form | 		return Form | ||||||
| @ -69,6 +96,8 @@ func Default(method, contentType string) Binding { | |||||||
| 		return ProtoBuf | 		return ProtoBuf | ||||||
| 	case MIMEMSGPACK, MIMEMSGPACK2: | 	case MIMEMSGPACK, MIMEMSGPACK2: | ||||||
| 		return MsgPack | 		return MsgPack | ||||||
|  | 	case MIMEYAML: | ||||||
|  | 		return YAML | ||||||
| 	default: //case MIMEPOSTForm, MIMEMultipartPOSTForm: | 	default: //case MIMEPOSTForm, MIMEMultipartPOSTForm: | ||||||
| 		return Form | 		return Form | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										72
									
								
								binding/binding_body_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										72
									
								
								binding/binding_body_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,72 @@ | |||||||
|  | package binding | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin/testdata/protoexample" | ||||||
|  | 	"github.com/golang/protobuf/proto" | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | 	"github.com/ugorji/go/codec" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestBindingBody(t *testing.T) { | ||||||
|  | 	for _, tt := range []struct { | ||||||
|  | 		name    string | ||||||
|  | 		binding BindingBody | ||||||
|  | 		body    string | ||||||
|  | 		want    string | ||||||
|  | 	}{ | ||||||
|  | 		{ | ||||||
|  | 			name:    "JSON binding", | ||||||
|  | 			binding: JSON, | ||||||
|  | 			body:    `{"foo":"FOO"}`, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name:    "XML binding", | ||||||
|  | 			binding: XML, | ||||||
|  | 			body: `<?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <root> | ||||||
|  |    <foo>FOO</foo> | ||||||
|  | </root>`, | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name:    "MsgPack binding", | ||||||
|  | 			binding: MsgPack, | ||||||
|  | 			body:    msgPackBody(t), | ||||||
|  | 		}, | ||||||
|  | 		{ | ||||||
|  | 			name:    "YAML binding", | ||||||
|  | 			binding: YAML, | ||||||
|  | 			body:    `foo: FOO`, | ||||||
|  | 		}, | ||||||
|  | 	} { | ||||||
|  | 		t.Logf("testing: %s", tt.name) | ||||||
|  | 		req := requestWithBody("POST", "/", tt.body) | ||||||
|  | 		form := FooStruct{} | ||||||
|  | 		body, _ := ioutil.ReadAll(req.Body) | ||||||
|  | 		assert.NoError(t, tt.binding.BindBody(body, &form)) | ||||||
|  | 		assert.Equal(t, FooStruct{"FOO"}, form) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func msgPackBody(t *testing.T) string { | ||||||
|  | 	test := FooStruct{"FOO"} | ||||||
|  | 	h := new(codec.MsgpackHandle) | ||||||
|  | 	buf := bytes.NewBuffer(nil) | ||||||
|  | 	assert.NoError(t, codec.NewEncoder(buf, h).Encode(test)) | ||||||
|  | 	return buf.String() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindingBodyProto(t *testing.T) { | ||||||
|  | 	test := protoexample.Test{ | ||||||
|  | 		Label: proto.String("FOO"), | ||||||
|  | 	} | ||||||
|  | 	data, _ := proto.Marshal(&test) | ||||||
|  | 	req := requestWithBody("POST", "/", string(data)) | ||||||
|  | 	form := protoexample.Test{} | ||||||
|  | 	body, _ := ioutil.ReadAll(req.Body) | ||||||
|  | 	assert.NoError(t, ProtoBuf.BindBody(body, &form)) | ||||||
|  | 	assert.Equal(t, test, form) | ||||||
|  | } | ||||||
| @ -11,10 +11,11 @@ import ( | |||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"mime/multipart" | 	"mime/multipart" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"strconv" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin/binding/example" | 	"github.com/gin-gonic/gin/testdata/protoexample" | ||||||
| 	"github.com/golang/protobuf/proto" | 	"github.com/golang/protobuf/proto" | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| 	"github.com/ugorji/go/codec" | 	"github.com/ugorji/go/codec" | ||||||
| @ -29,6 +30,11 @@ type FooBarStruct struct { | |||||||
| 	Bar string `msgpack:"bar" json:"bar" form:"bar" xml:"bar" binding:"required"` | 	Bar string `msgpack:"bar" json:"bar" form:"bar" xml:"bar" binding:"required"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type FooDefaultBarStruct struct { | ||||||
|  | 	FooStruct | ||||||
|  | 	Bar string `msgpack:"bar" json:"bar" form:"bar,default=hello" xml:"bar" binding:"required"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type FooStructUseNumber struct { | type FooStructUseNumber struct { | ||||||
| 	Foo interface{} `json:"foo" binding:"required"` | 	Foo interface{} `json:"foo" binding:"required"` | ||||||
| } | } | ||||||
| @ -69,11 +75,27 @@ type FooStructForSliceType struct { | |||||||
| 	SliceFoo []int `form:"slice_foo"` | 	SliceFoo []int `form:"slice_foo"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type FooStructForStructType struct { | ||||||
|  | 	StructFoo struct { | ||||||
|  | 		Idx int `form:"idx"` | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type FooStructForStructPointerType struct { | ||||||
|  | 	StructPointerFoo *struct { | ||||||
|  | 		Name string `form:"name"` | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type FooStructForSliceMapType struct { | type FooStructForSliceMapType struct { | ||||||
| 	// Unknown type: not support map | 	// Unknown type: not support map | ||||||
| 	SliceMapFoo []map[string]interface{} `form:"slice_map_foo"` | 	SliceMapFoo []map[string]interface{} `form:"slice_map_foo"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type FooStructForBoolType struct { | ||||||
|  | 	BoolFoo bool `form:"bool_foo"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| type FooBarStructForIntType struct { | type FooBarStructForIntType struct { | ||||||
| 	IntFoo int `form:"int_foo"` | 	IntFoo int `form:"int_foo"` | ||||||
| 	IntBar int `form:"int_bar" binding:"required"` | 	IntBar int `form:"int_bar" binding:"required"` | ||||||
| @ -139,27 +161,46 @@ type FooBarStructForFloat64Type struct { | |||||||
| 	Float64Bar float64 `form:"float64_bar" binding:"required"` | 	Float64Bar float64 `form:"float64_bar" binding:"required"` | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | type FooStructForStringPtrType struct { | ||||||
|  | 	PtrFoo *string `form:"ptr_foo"` | ||||||
|  | 	PtrBar *string `form:"ptr_bar" binding:"required"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type FooStructForMapPtrType struct { | ||||||
|  | 	PtrBar *map[string]interface{} `form:"ptr_bar"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestBindingDefault(t *testing.T) { | func TestBindingDefault(t *testing.T) { | ||||||
| 	assert.Equal(t, Default("GET", ""), Form) | 	assert.Equal(t, Form, Default("GET", "")) | ||||||
| 	assert.Equal(t, Default("GET", MIMEJSON), Form) | 	assert.Equal(t, Form, Default("GET", MIMEJSON)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEJSON), JSON) | 	assert.Equal(t, JSON, Default("POST", MIMEJSON)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEJSON), JSON) | 	assert.Equal(t, JSON, Default("PUT", MIMEJSON)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEXML), XML) | 	assert.Equal(t, XML, Default("POST", MIMEXML)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEXML2), XML) | 	assert.Equal(t, XML, Default("PUT", MIMEXML2)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEPOSTForm), Form) | 	assert.Equal(t, Form, Default("POST", MIMEPOSTForm)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEPOSTForm), Form) | 	assert.Equal(t, Form, Default("PUT", MIMEPOSTForm)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEMultipartPOSTForm), Form) | 	assert.Equal(t, Form, Default("POST", MIMEMultipartPOSTForm)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEMultipartPOSTForm), Form) | 	assert.Equal(t, Form, Default("PUT", MIMEMultipartPOSTForm)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEPROTOBUF), ProtoBuf) | 	assert.Equal(t, ProtoBuf, Default("POST", MIMEPROTOBUF)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEPROTOBUF), ProtoBuf) | 	assert.Equal(t, ProtoBuf, Default("PUT", MIMEPROTOBUF)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, Default("POST", MIMEMSGPACK), MsgPack) | 	assert.Equal(t, MsgPack, Default("POST", MIMEMSGPACK)) | ||||||
| 	assert.Equal(t, Default("PUT", MIMEMSGPACK2), MsgPack) | 	assert.Equal(t, MsgPack, Default("PUT", MIMEMSGPACK2)) | ||||||
|  | 
 | ||||||
|  | 	assert.Equal(t, YAML, Default("POST", MIMEYAML)) | ||||||
|  | 	assert.Equal(t, YAML, Default("PUT", MIMEYAML)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindingJSONNilBody(t *testing.T) { | ||||||
|  | 	var obj FooStruct | ||||||
|  | 	req, _ := http.NewRequest(http.MethodPost, "/", nil) | ||||||
|  | 	err := JSON.Bind(req, &obj) | ||||||
|  | 	assert.Error(t, err) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingJSON(t *testing.T) { | func TestBindingJSON(t *testing.T) { | ||||||
| @ -195,6 +236,18 @@ func TestBindingForm2(t *testing.T) { | |||||||
| 		"", "") | 		"", "") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestBindingFormDefaultValue(t *testing.T) { | ||||||
|  | 	testFormBindingDefaultValue(t, "POST", | ||||||
|  | 		"/", "/", | ||||||
|  | 		"foo=bar", "bar2=foo") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindingFormDefaultValue2(t *testing.T) { | ||||||
|  | 	testFormBindingDefaultValue(t, "GET", | ||||||
|  | 		"/?foo=bar", "/?bar2=foo", | ||||||
|  | 		"", "") | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestBindingFormForTime(t *testing.T) { | func TestBindingFormForTime(t *testing.T) { | ||||||
| 	testFormBindingForTime(t, "POST", | 	testFormBindingForTime(t, "POST", | ||||||
| 		"/", "/", | 		"/", "/", | ||||||
| @ -361,6 +414,30 @@ func TestBindingFormForType(t *testing.T) { | |||||||
| 	testFormBindingForType(t, "GET", | 	testFormBindingForType(t, "GET", | ||||||
| 		"/?float64_foo=&float64_bar=-12.34", "/?bar2=12.3", | 		"/?float64_foo=&float64_bar=-12.34", "/?bar2=12.3", | ||||||
| 		"", "", "Float64") | 		"", "", "Float64") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "POST", | ||||||
|  | 		"/", "/", | ||||||
|  | 		"ptr_bar=test", "bar2=test", "Ptr") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "GET", | ||||||
|  | 		"/?ptr_bar=test", "/?bar2=test", | ||||||
|  | 		"", "", "Ptr") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "POST", | ||||||
|  | 		"/", "/", | ||||||
|  | 		"idx=123", "id1=1", "Struct") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "GET", | ||||||
|  | 		"/?idx=123", "/?id1=1", | ||||||
|  | 		"", "", "Struct") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "POST", | ||||||
|  | 		"/", "/", | ||||||
|  | 		"name=thinkerou", "name1=ou", "StructPointer") | ||||||
|  | 
 | ||||||
|  | 	testFormBindingForType(t, "GET", | ||||||
|  | 		"/?name=thinkerou", "/?name1=ou", | ||||||
|  | 		"", "", "StructPointer") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingQuery(t *testing.T) { | func TestBindingQuery(t *testing.T) { | ||||||
| @ -387,6 +464,12 @@ func TestBindingQueryFail2(t *testing.T) { | |||||||
| 		"map_foo=unused", "") | 		"map_foo=unused", "") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestBindingQueryBoolFail(t *testing.T) { | ||||||
|  | 	testQueryBindingBoolFail(t, "GET", | ||||||
|  | 		"/?bool_foo=fasl", "/?bar2=foo", | ||||||
|  | 		"bool_foo=unused", "") | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestBindingXML(t *testing.T) { | func TestBindingXML(t *testing.T) { | ||||||
| 	testBodyBinding(t, | 	testBodyBinding(t, | ||||||
| 		XML, "xml", | 		XML, "xml", | ||||||
| @ -401,40 +484,60 @@ func TestBindingXMLFail(t *testing.T) { | |||||||
| 		"<map><foo>bar<foo></map>", "<map><bar>foo</bar></map>") | 		"<map><foo>bar<foo></map>", "<map><bar>foo</bar></map>") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestBindingYAML(t *testing.T) { | ||||||
|  | 	testBodyBinding(t, | ||||||
|  | 		YAML, "yaml", | ||||||
|  | 		"/", "/", | ||||||
|  | 		`foo: bar`, `bar: foo`) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindingYAMLFail(t *testing.T) { | ||||||
|  | 	testBodyBindingFail(t, | ||||||
|  | 		YAML, "yaml", | ||||||
|  | 		"/", "/", | ||||||
|  | 		`foo:\nbar`, `bar: foo`) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func createFormPostRequest() *http.Request { | func createFormPostRequest() *http.Request { | ||||||
| 	req, _ := http.NewRequest("POST", "/?foo=getfoo&bar=getbar", bytes.NewBufferString("foo=bar&bar=foo")) | 	req, _ := http.NewRequest("POST", "/?foo=getfoo&bar=getbar", bytes.NewBufferString("foo=bar&bar=foo")) | ||||||
| 	req.Header.Set("Content-Type", MIMEPOSTForm) | 	req.Header.Set("Content-Type", MIMEPOSTForm) | ||||||
| 	return req | 	return req | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func createDefaultFormPostRequest() *http.Request { | ||||||
|  | 	req, _ := http.NewRequest("POST", "/?foo=getfoo&bar=getbar", bytes.NewBufferString("foo=bar")) | ||||||
|  | 	req.Header.Set("Content-Type", MIMEPOSTForm) | ||||||
|  | 	return req | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func createFormPostRequestFail() *http.Request { | func createFormPostRequestFail() *http.Request { | ||||||
| 	req, _ := http.NewRequest("POST", "/?map_foo=getfoo", bytes.NewBufferString("map_foo=bar")) | 	req, _ := http.NewRequest("POST", "/?map_foo=getfoo", bytes.NewBufferString("map_foo=bar")) | ||||||
| 	req.Header.Set("Content-Type", MIMEPOSTForm) | 	req.Header.Set("Content-Type", MIMEPOSTForm) | ||||||
| 	return req | 	return req | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func createFormMultipartRequest() *http.Request { | func createFormMultipartRequest(t *testing.T) *http.Request { | ||||||
| 	boundary := "--testboundary" | 	boundary := "--testboundary" | ||||||
| 	body := new(bytes.Buffer) | 	body := new(bytes.Buffer) | ||||||
| 	mw := multipart.NewWriter(body) | 	mw := multipart.NewWriter(body) | ||||||
| 	defer mw.Close() | 	defer mw.Close() | ||||||
| 
 | 
 | ||||||
| 	mw.SetBoundary(boundary) | 	assert.NoError(t, mw.SetBoundary(boundary)) | ||||||
| 	mw.WriteField("foo", "bar") | 	assert.NoError(t, mw.WriteField("foo", "bar")) | ||||||
| 	mw.WriteField("bar", "foo") | 	assert.NoError(t, mw.WriteField("bar", "foo")) | ||||||
| 	req, _ := http.NewRequest("POST", "/?foo=getfoo&bar=getbar", body) | 	req, _ := http.NewRequest("POST", "/?foo=getfoo&bar=getbar", body) | ||||||
| 	req.Header.Set("Content-Type", MIMEMultipartPOSTForm+"; boundary="+boundary) | 	req.Header.Set("Content-Type", MIMEMultipartPOSTForm+"; boundary="+boundary) | ||||||
| 	return req | 	return req | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func createFormMultipartRequestFail() *http.Request { | func createFormMultipartRequestFail(t *testing.T) *http.Request { | ||||||
| 	boundary := "--testboundary" | 	boundary := "--testboundary" | ||||||
| 	body := new(bytes.Buffer) | 	body := new(bytes.Buffer) | ||||||
| 	mw := multipart.NewWriter(body) | 	mw := multipart.NewWriter(body) | ||||||
| 	defer mw.Close() | 	defer mw.Close() | ||||||
| 
 | 
 | ||||||
| 	mw.SetBoundary(boundary) | 	assert.NoError(t, mw.SetBoundary(boundary)) | ||||||
| 	mw.WriteField("map_foo", "bar") | 	assert.NoError(t, mw.WriteField("map_foo", "bar")) | ||||||
| 	req, _ := http.NewRequest("POST", "/?map_foo=getfoo", body) | 	req, _ := http.NewRequest("POST", "/?map_foo=getfoo", body) | ||||||
| 	req.Header.Set("Content-Type", MIMEMultipartPOSTForm+"; boundary="+boundary) | 	req.Header.Set("Content-Type", MIMEMultipartPOSTForm+"; boundary="+boundary) | ||||||
| 	return req | 	return req | ||||||
| @ -443,11 +546,20 @@ func createFormMultipartRequestFail() *http.Request { | |||||||
| func TestBindingFormPost(t *testing.T) { | func TestBindingFormPost(t *testing.T) { | ||||||
| 	req := createFormPostRequest() | 	req := createFormPostRequest() | ||||||
| 	var obj FooBarStruct | 	var obj FooBarStruct | ||||||
| 	FormPost.Bind(req, &obj) | 	assert.NoError(t, FormPost.Bind(req, &obj)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, FormPost.Name(), "form-urlencoded") | 	assert.Equal(t, "form-urlencoded", FormPost.Name()) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 	assert.Equal(t, obj.Bar, "foo") | 	assert.Equal(t, "foo", obj.Bar) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindingDefaultValueFormPost(t *testing.T) { | ||||||
|  | 	req := createDefaultFormPostRequest() | ||||||
|  | 	var obj FooDefaultBarStruct | ||||||
|  | 	assert.NoError(t, FormPost.Bind(req, &obj)) | ||||||
|  | 
 | ||||||
|  | 	assert.Equal(t, "bar", obj.Foo) | ||||||
|  | 	assert.Equal(t, "hello", obj.Bar) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingFormPostFail(t *testing.T) { | func TestBindingFormPostFail(t *testing.T) { | ||||||
| @ -458,24 +570,24 @@ func TestBindingFormPostFail(t *testing.T) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingFormMultipart(t *testing.T) { | func TestBindingFormMultipart(t *testing.T) { | ||||||
| 	req := createFormMultipartRequest() | 	req := createFormMultipartRequest(t) | ||||||
| 	var obj FooBarStruct | 	var obj FooBarStruct | ||||||
| 	FormMultipart.Bind(req, &obj) | 	assert.NoError(t, FormMultipart.Bind(req, &obj)) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, FormMultipart.Name(), "multipart/form-data") | 	assert.Equal(t, "multipart/form-data", FormMultipart.Name()) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 	assert.Equal(t, obj.Bar, "foo") | 	assert.Equal(t, "foo", obj.Bar) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingFormMultipartFail(t *testing.T) { | func TestBindingFormMultipartFail(t *testing.T) { | ||||||
| 	req := createFormMultipartRequestFail() | 	req := createFormMultipartRequestFail(t) | ||||||
| 	var obj FooStructForMapType | 	var obj FooStructForMapType | ||||||
| 	err := FormMultipart.Bind(req, &obj) | 	err := FormMultipart.Bind(req, &obj) | ||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingProtoBuf(t *testing.T) { | func TestBindingProtoBuf(t *testing.T) { | ||||||
| 	test := &example.Test{ | 	test := &protoexample.Test{ | ||||||
| 		Label: proto.String("yes"), | 		Label: proto.String("yes"), | ||||||
| 	} | 	} | ||||||
| 	data, _ := proto.Marshal(test) | 	data, _ := proto.Marshal(test) | ||||||
| @ -487,7 +599,7 @@ func TestBindingProtoBuf(t *testing.T) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestBindingProtoBufFail(t *testing.T) { | func TestBindingProtoBufFail(t *testing.T) { | ||||||
| 	test := &example.Test{ | 	test := &protoexample.Test{ | ||||||
| 		Label: proto.String("yes"), | 		Label: proto.String("yes"), | ||||||
| 	} | 	} | ||||||
| 	data, _ := proto.Marshal(test) | 	data, _ := proto.Marshal(test) | ||||||
| @ -558,9 +670,52 @@ func TestExistsFails(t *testing.T) { | |||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestUriBinding(t *testing.T) { | ||||||
|  | 	b := Uri | ||||||
|  | 	assert.Equal(t, "uri", b.Name()) | ||||||
|  | 
 | ||||||
|  | 	type Tag struct { | ||||||
|  | 		Name string `uri:"name"` | ||||||
|  | 	} | ||||||
|  | 	var tag Tag | ||||||
|  | 	m := make(map[string][]string) | ||||||
|  | 	m["name"] = []string{"thinkerou"} | ||||||
|  | 	assert.NoError(t, b.BindUri(m, &tag)) | ||||||
|  | 	assert.Equal(t, "thinkerou", tag.Name) | ||||||
|  | 
 | ||||||
|  | 	type NotSupportStruct struct { | ||||||
|  | 		Name map[string]interface{} `uri:"name"` | ||||||
|  | 	} | ||||||
|  | 	var not NotSupportStruct | ||||||
|  | 	assert.Error(t, b.BindUri(m, ¬)) | ||||||
|  | 	assert.Equal(t, map[string]interface{}(nil), not.Name) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestUriInnerBinding(t *testing.T) { | ||||||
|  | 	type Tag struct { | ||||||
|  | 		Name string `uri:"name"` | ||||||
|  | 		S    struct { | ||||||
|  | 			Age int `uri:"age"` | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	expectedName := "mike" | ||||||
|  | 	expectedAge := 25 | ||||||
|  | 
 | ||||||
|  | 	m := map[string][]string{ | ||||||
|  | 		"name": {expectedName}, | ||||||
|  | 		"age":  {strconv.Itoa(expectedAge)}, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var tag Tag | ||||||
|  | 	assert.NoError(t, Uri.BindUri(m, &tag)) | ||||||
|  | 	assert.Equal(t, tag.Name, expectedName) | ||||||
|  | 	assert.Equal(t, tag.S.Age, expectedAge) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func testFormBinding(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBinding(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStruct{} | 	obj := FooBarStruct{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -569,8 +724,8 @@ func testFormBinding(t *testing.T, method, path, badPath, body, badBody string) | |||||||
| 	} | 	} | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 	assert.Equal(t, obj.Bar, "foo") | 	assert.Equal(t, "foo", obj.Bar) | ||||||
| 
 | 
 | ||||||
| 	obj = FooBarStruct{} | 	obj = FooBarStruct{} | ||||||
| 	req = requestWithBody(method, badPath, badBody) | 	req = requestWithBody(method, badPath, badBody) | ||||||
| @ -578,9 +733,29 @@ func testFormBinding(t *testing.T, method, path, badPath, body, badBody string) | |||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func testFormBindingDefaultValue(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
|  | 	b := Form | ||||||
|  | 	assert.Equal(t, "form", b.Name()) | ||||||
|  | 
 | ||||||
|  | 	obj := FooDefaultBarStruct{} | ||||||
|  | 	req := requestWithBody(method, path, body) | ||||||
|  | 	if method == "POST" { | ||||||
|  | 		req.Header.Add("Content-Type", MIMEPOSTForm) | ||||||
|  | 	} | ||||||
|  | 	err := b.Bind(req, &obj) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, "bar", obj.Foo) | ||||||
|  | 	assert.Equal(t, "hello", obj.Bar) | ||||||
|  | 
 | ||||||
|  | 	obj = FooDefaultBarStruct{} | ||||||
|  | 	req = requestWithBody(method, badPath, badBody) | ||||||
|  | 	err = JSON.Bind(req, &obj) | ||||||
|  | 	assert.Error(t, err) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestFormBindingFail(t *testing.T) { | func TestFormBindingFail(t *testing.T) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStruct{} | 	obj := FooBarStruct{} | ||||||
| 	req, _ := http.NewRequest("POST", "/", nil) | 	req, _ := http.NewRequest("POST", "/", nil) | ||||||
| @ -590,7 +765,7 @@ func TestFormBindingFail(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| func TestFormPostBindingFail(t *testing.T) { | func TestFormPostBindingFail(t *testing.T) { | ||||||
| 	b := FormPost | 	b := FormPost | ||||||
| 	assert.Equal(t, b.Name(), "form-urlencoded") | 	assert.Equal(t, "form-urlencoded", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStruct{} | 	obj := FooBarStruct{} | ||||||
| 	req, _ := http.NewRequest("POST", "/", nil) | 	req, _ := http.NewRequest("POST", "/", nil) | ||||||
| @ -600,7 +775,7 @@ func TestFormPostBindingFail(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| func TestFormMultipartBindingFail(t *testing.T) { | func TestFormMultipartBindingFail(t *testing.T) { | ||||||
| 	b := FormMultipart | 	b := FormMultipart | ||||||
| 	assert.Equal(t, b.Name(), "multipart/form-data") | 	assert.Equal(t, "multipart/form-data", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStruct{} | 	obj := FooBarStruct{} | ||||||
| 	req, _ := http.NewRequest("POST", "/", nil) | 	req, _ := http.NewRequest("POST", "/", nil) | ||||||
| @ -610,7 +785,7 @@ func TestFormMultipartBindingFail(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| func testFormBindingForTime(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingForTime(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStructForTimeType{} | 	obj := FooBarStructForTimeType{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -620,10 +795,10 @@ func testFormBindingForTime(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 
 | 
 | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.TimeFoo.Unix(), int64(1510675200)) | 	assert.Equal(t, int64(1510675200), obj.TimeFoo.Unix()) | ||||||
| 	assert.Equal(t, obj.TimeFoo.Location().String(), "Asia/Chongqing") | 	assert.Equal(t, "Asia/Chongqing", obj.TimeFoo.Location().String()) | ||||||
| 	assert.Equal(t, obj.TimeBar.Unix(), int64(-62135596800)) | 	assert.Equal(t, int64(-62135596800), obj.TimeBar.Unix()) | ||||||
| 	assert.Equal(t, obj.TimeBar.Location().String(), "UTC") | 	assert.Equal(t, "UTC", obj.TimeBar.Location().String()) | ||||||
| 
 | 
 | ||||||
| 	obj = FooBarStructForTimeType{} | 	obj = FooBarStructForTimeType{} | ||||||
| 	req = requestWithBody(method, badPath, badBody) | 	req = requestWithBody(method, badPath, badBody) | ||||||
| @ -633,7 +808,7 @@ func testFormBindingForTime(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 
 | 
 | ||||||
| func testFormBindingForTimeNotFormat(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingForTimeNotFormat(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructForTimeTypeNotFormat{} | 	obj := FooStructForTimeTypeNotFormat{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -651,7 +826,7 @@ func testFormBindingForTimeNotFormat(t *testing.T, method, path, badPath, body, | |||||||
| 
 | 
 | ||||||
| func testFormBindingForTimeFailFormat(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingForTimeFailFormat(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructForTimeTypeFailFormat{} | 	obj := FooStructForTimeTypeFailFormat{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -669,7 +844,7 @@ func testFormBindingForTimeFailFormat(t *testing.T, method, path, badPath, body, | |||||||
| 
 | 
 | ||||||
| func testFormBindingForTimeFailLocation(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingForTimeFailLocation(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructForTimeTypeFailLocation{} | 	obj := FooStructForTimeTypeFailLocation{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -687,7 +862,7 @@ func testFormBindingForTimeFailLocation(t *testing.T, method, path, badPath, bod | |||||||
| 
 | 
 | ||||||
| func testFormBindingInvalidName(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingInvalidName(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := InvalidNameType{} | 	obj := InvalidNameType{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -696,7 +871,7 @@ func testFormBindingInvalidName(t *testing.T, method, path, badPath, body, badBo | |||||||
| 	} | 	} | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.TestName, "") | 	assert.Equal(t, "", obj.TestName) | ||||||
| 
 | 
 | ||||||
| 	obj = InvalidNameType{} | 	obj = InvalidNameType{} | ||||||
| 	req = requestWithBody(method, badPath, badBody) | 	req = requestWithBody(method, badPath, badBody) | ||||||
| @ -706,7 +881,7 @@ func testFormBindingInvalidName(t *testing.T, method, path, badPath, body, badBo | |||||||
| 
 | 
 | ||||||
| func testFormBindingInvalidName2(t *testing.T, method, path, badPath, body, badBody string) { | func testFormBindingInvalidName2(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := InvalidNameMapType{} | 	obj := InvalidNameMapType{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -724,7 +899,7 @@ func testFormBindingInvalidName2(t *testing.T, method, path, badPath, body, badB | |||||||
| 
 | 
 | ||||||
| func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody string, typ string) { | func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody string, typ string) { | ||||||
| 	b := Form | 	b := Form | ||||||
| 	assert.Equal(t, b.Name(), "form") | 	assert.Equal(t, "form", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| 	if method == "POST" { | 	if method == "POST" { | ||||||
| @ -735,8 +910,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForIntType{} | 		obj := FooBarStructForIntType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.IntFoo, int(0)) | 		assert.Equal(t, int(0), obj.IntFoo) | ||||||
| 		assert.Equal(t, obj.IntBar, int(-12)) | 		assert.Equal(t, int(-12), obj.IntBar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForIntType{} | 		obj = FooBarStructForIntType{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -746,8 +921,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForInt8Type{} | 		obj := FooBarStructForInt8Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Int8Foo, int8(0)) | 		assert.Equal(t, int8(0), obj.Int8Foo) | ||||||
| 		assert.Equal(t, obj.Int8Bar, int8(-12)) | 		assert.Equal(t, int8(-12), obj.Int8Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForInt8Type{} | 		obj = FooBarStructForInt8Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -757,8 +932,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForInt16Type{} | 		obj := FooBarStructForInt16Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Int16Foo, int16(0)) | 		assert.Equal(t, int16(0), obj.Int16Foo) | ||||||
| 		assert.Equal(t, obj.Int16Bar, int16(-12)) | 		assert.Equal(t, int16(-12), obj.Int16Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForInt16Type{} | 		obj = FooBarStructForInt16Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -768,8 +943,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForInt32Type{} | 		obj := FooBarStructForInt32Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Int32Foo, int32(0)) | 		assert.Equal(t, int32(0), obj.Int32Foo) | ||||||
| 		assert.Equal(t, obj.Int32Bar, int32(-12)) | 		assert.Equal(t, int32(-12), obj.Int32Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForInt32Type{} | 		obj = FooBarStructForInt32Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -779,8 +954,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForInt64Type{} | 		obj := FooBarStructForInt64Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Int64Foo, int64(0)) | 		assert.Equal(t, int64(0), obj.Int64Foo) | ||||||
| 		assert.Equal(t, obj.Int64Bar, int64(-12)) | 		assert.Equal(t, int64(-12), obj.Int64Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForInt64Type{} | 		obj = FooBarStructForInt64Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -790,8 +965,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForUintType{} | 		obj := FooBarStructForUintType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.UintFoo, uint(0x0)) | 		assert.Equal(t, uint(0x0), obj.UintFoo) | ||||||
| 		assert.Equal(t, obj.UintBar, uint(0xc)) | 		assert.Equal(t, uint(0xc), obj.UintBar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForUintType{} | 		obj = FooBarStructForUintType{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -801,8 +976,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForUint8Type{} | 		obj := FooBarStructForUint8Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Uint8Foo, uint8(0x0)) | 		assert.Equal(t, uint8(0x0), obj.Uint8Foo) | ||||||
| 		assert.Equal(t, obj.Uint8Bar, uint8(0xc)) | 		assert.Equal(t, uint8(0xc), obj.Uint8Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForUint8Type{} | 		obj = FooBarStructForUint8Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -812,8 +987,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForUint16Type{} | 		obj := FooBarStructForUint16Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Uint16Foo, uint16(0x0)) | 		assert.Equal(t, uint16(0x0), obj.Uint16Foo) | ||||||
| 		assert.Equal(t, obj.Uint16Bar, uint16(0xc)) | 		assert.Equal(t, uint16(0xc), obj.Uint16Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForUint16Type{} | 		obj = FooBarStructForUint16Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -823,8 +998,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForUint32Type{} | 		obj := FooBarStructForUint32Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Uint32Foo, uint32(0x0)) | 		assert.Equal(t, uint32(0x0), obj.Uint32Foo) | ||||||
| 		assert.Equal(t, obj.Uint32Bar, uint32(0xc)) | 		assert.Equal(t, uint32(0xc), obj.Uint32Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForUint32Type{} | 		obj = FooBarStructForUint32Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -834,8 +1009,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForUint64Type{} | 		obj := FooBarStructForUint64Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Uint64Foo, uint64(0x0)) | 		assert.Equal(t, uint64(0x0), obj.Uint64Foo) | ||||||
| 		assert.Equal(t, obj.Uint64Bar, uint64(0xc)) | 		assert.Equal(t, uint64(0xc), obj.Uint64Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForUint64Type{} | 		obj = FooBarStructForUint64Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -845,8 +1020,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForFloat32Type{} | 		obj := FooBarStructForFloat32Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Float32Foo, float32(0.0)) | 		assert.Equal(t, float32(0.0), obj.Float32Foo) | ||||||
| 		assert.Equal(t, obj.Float32Bar, float32(-12.34)) | 		assert.Equal(t, float32(-12.34), obj.Float32Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForFloat32Type{} | 		obj = FooBarStructForFloat32Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -856,8 +1031,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForFloat64Type{} | 		obj := FooBarStructForFloat64Type{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.Float64Foo, float64(0.0)) | 		assert.Equal(t, float64(0.0), obj.Float64Foo) | ||||||
| 		assert.Equal(t, obj.Float64Bar, float64(-12.34)) | 		assert.Equal(t, float64(-12.34), obj.Float64Bar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForFloat64Type{} | 		obj = FooBarStructForFloat64Type{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -867,8 +1042,8 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooBarStructForBoolType{} | 		obj := FooBarStructForBoolType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.BoolFoo, false) | 		assert.False(t, obj.BoolFoo) | ||||||
| 		assert.Equal(t, obj.BoolBar, true) | 		assert.True(t, obj.BoolBar) | ||||||
| 
 | 
 | ||||||
| 		obj = FooBarStructForBoolType{} | 		obj = FooBarStructForBoolType{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| @ -878,12 +1053,34 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooStructForSliceType{} | 		obj := FooStructForSliceType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.NoError(t, err) | 		assert.NoError(t, err) | ||||||
| 		assert.Equal(t, obj.SliceFoo, []int{1, 2}) | 		assert.Equal(t, []int{1, 2}, obj.SliceFoo) | ||||||
| 
 | 
 | ||||||
| 		obj = FooStructForSliceType{} | 		obj = FooStructForSliceType{} | ||||||
| 		req = requestWithBody(method, badPath, badBody) | 		req = requestWithBody(method, badPath, badBody) | ||||||
| 		err = JSON.Bind(req, &obj) | 		err = JSON.Bind(req, &obj) | ||||||
| 		assert.Error(t, err) | 		assert.Error(t, err) | ||||||
|  | 	case "Struct": | ||||||
|  | 		obj := FooStructForStructType{} | ||||||
|  | 		err := b.Bind(req, &obj) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		assert.Equal(t, | ||||||
|  | 			struct { | ||||||
|  | 				Idx int "form:\"idx\"" | ||||||
|  | 			}(struct { | ||||||
|  | 				Idx int "form:\"idx\"" | ||||||
|  | 			}{Idx: 123}), | ||||||
|  | 			obj.StructFoo) | ||||||
|  | 	case "StructPointer": | ||||||
|  | 		obj := FooStructForStructPointerType{} | ||||||
|  | 		err := b.Bind(req, &obj) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		assert.Equal(t, | ||||||
|  | 			struct { | ||||||
|  | 				Name string "form:\"name\"" | ||||||
|  | 			}(struct { | ||||||
|  | 				Name string "form:\"name\"" | ||||||
|  | 			}{Name: "thinkerou"}), | ||||||
|  | 			*obj.StructPointerFoo) | ||||||
| 	case "Map": | 	case "Map": | ||||||
| 		obj := FooStructForMapType{} | 		obj := FooStructForMapType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| @ -892,12 +1089,33 @@ func testFormBindingForType(t *testing.T, method, path, badPath, body, badBody s | |||||||
| 		obj := FooStructForSliceMapType{} | 		obj := FooStructForSliceMapType{} | ||||||
| 		err := b.Bind(req, &obj) | 		err := b.Bind(req, &obj) | ||||||
| 		assert.Error(t, err) | 		assert.Error(t, err) | ||||||
|  | 	case "Ptr": | ||||||
|  | 		obj := FooStructForStringPtrType{} | ||||||
|  | 		err := b.Bind(req, &obj) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		assert.Nil(t, obj.PtrFoo) | ||||||
|  | 		assert.Equal(t, "test", *obj.PtrBar) | ||||||
|  | 
 | ||||||
|  | 		obj = FooStructForStringPtrType{} | ||||||
|  | 		obj.PtrBar = new(string) | ||||||
|  | 		err = b.Bind(req, &obj) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		assert.Equal(t, "test", *obj.PtrBar) | ||||||
|  | 
 | ||||||
|  | 		objErr := FooStructForMapPtrType{} | ||||||
|  | 		err = b.Bind(req, &objErr) | ||||||
|  | 		assert.Error(t, err) | ||||||
|  | 
 | ||||||
|  | 		obj = FooStructForStringPtrType{} | ||||||
|  | 		req = requestWithBody(method, badPath, badBody) | ||||||
|  | 		err = b.Bind(req, &obj) | ||||||
|  | 		assert.Error(t, err) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testQueryBinding(t *testing.T, method, path, badPath, body, badBody string) { | func testQueryBinding(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Query | 	b := Query | ||||||
| 	assert.Equal(t, b.Name(), "query") | 	assert.Equal(t, "query", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooBarStruct{} | 	obj := FooBarStruct{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -906,13 +1124,13 @@ func testQueryBinding(t *testing.T, method, path, badPath, body, badBody string) | |||||||
| 	} | 	} | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 	assert.Equal(t, obj.Bar, "foo") | 	assert.Equal(t, "foo", obj.Bar) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testQueryBindingFail(t *testing.T, method, path, badPath, body, badBody string) { | func testQueryBindingFail(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
| 	b := Query | 	b := Query | ||||||
| 	assert.Equal(t, b.Name(), "query") | 	assert.Equal(t, "query", b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructForMapType{} | 	obj := FooStructForMapType{} | ||||||
| 	req := requestWithBody(method, path, body) | 	req := requestWithBody(method, path, body) | ||||||
| @ -923,14 +1141,27 @@ func testQueryBindingFail(t *testing.T, method, path, badPath, body, badBody str | |||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func testQueryBindingBoolFail(t *testing.T, method, path, badPath, body, badBody string) { | ||||||
|  | 	b := Query | ||||||
|  | 	assert.Equal(t, "query", b.Name()) | ||||||
|  | 
 | ||||||
|  | 	obj := FooStructForBoolType{} | ||||||
|  | 	req := requestWithBody(method, path, body) | ||||||
|  | 	if method == "POST" { | ||||||
|  | 		req.Header.Add("Content-Type", MIMEPOSTForm) | ||||||
|  | 	} | ||||||
|  | 	err := b.Bind(req, &obj) | ||||||
|  | 	assert.Error(t, err) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func testBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStruct{} | 	obj := FooStruct{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 
 | 
 | ||||||
| 	obj = FooStruct{} | 	obj = FooStruct{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| @ -939,7 +1170,7 @@ func testBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testBodyBindingUseNumber(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testBodyBindingUseNumber(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructUseNumber{} | 	obj := FooStructUseNumber{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| @ -949,7 +1180,7 @@ func testBodyBindingUseNumber(t *testing.T, b Binding, name, path, badPath, body | |||||||
| 	// we hope it is int64(123) | 	// we hope it is int64(123) | ||||||
| 	v, e := obj.Foo.(json.Number).Int64() | 	v, e := obj.Foo.(json.Number).Int64() | ||||||
| 	assert.NoError(t, e) | 	assert.NoError(t, e) | ||||||
| 	assert.Equal(t, v, int64(123)) | 	assert.Equal(t, int64(123), v) | ||||||
| 
 | 
 | ||||||
| 	obj = FooStructUseNumber{} | 	obj = FooStructUseNumber{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| @ -958,7 +1189,7 @@ func testBodyBindingUseNumber(t *testing.T, b Binding, name, path, badPath, body | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testBodyBindingUseNumber2(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testBodyBindingUseNumber2(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStructUseNumber{} | 	obj := FooStructUseNumber{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| @ -967,7 +1198,7 @@ func testBodyBindingUseNumber2(t *testing.T, b Binding, name, path, badPath, bod | |||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	// it will return float64(123) if not use EnableDecoderUseNumber | 	// it will return float64(123) if not use EnableDecoderUseNumber | ||||||
| 	// maybe it is not hoped | 	// maybe it is not hoped | ||||||
| 	assert.Equal(t, obj.Foo, float64(123)) | 	assert.Equal(t, float64(123), obj.Foo) | ||||||
| 
 | 
 | ||||||
| 	obj = FooStructUseNumber{} | 	obj = FooStructUseNumber{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| @ -976,13 +1207,13 @@ func testBodyBindingUseNumber2(t *testing.T, b Binding, name, path, badPath, bod | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStruct{} | 	obj := FooStruct{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| 	assert.Equal(t, obj.Foo, "") | 	assert.Equal(t, "", obj.Foo) | ||||||
| 
 | 
 | ||||||
| 	obj = FooStruct{} | 	obj = FooStruct{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| @ -991,16 +1222,16 @@ func testBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, bad | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testProtoBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testProtoBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := example.Test{} | 	obj := protoexample.Test{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| 	req.Header.Add("Content-Type", MIMEPROTOBUF) | 	req.Header.Add("Content-Type", MIMEPROTOBUF) | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, *obj.Label, "yes") | 	assert.Equal(t, "yes", *obj.Label) | ||||||
| 
 | 
 | ||||||
| 	obj = example.Test{} | 	obj = protoexample.Test{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| 	req.Header.Add("Content-Type", MIMEPROTOBUF) | 	req.Header.Add("Content-Type", MIMEPROTOBUF) | ||||||
| 	err = ProtoBuf.Bind(req, &obj) | 	err = ProtoBuf.Bind(req, &obj) | ||||||
| @ -1014,9 +1245,9 @@ func (h hook) Read([]byte) (int, error) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testProtoBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testProtoBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := example.Test{} | 	obj := protoexample.Test{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| 
 | 
 | ||||||
| 	req.Body = ioutil.NopCloser(&hook{}) | 	req.Body = ioutil.NopCloser(&hook{}) | ||||||
| @ -1024,7 +1255,7 @@ func testProtoBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body | |||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.Error(t, err) | 	assert.Error(t, err) | ||||||
| 
 | 
 | ||||||
| 	obj = example.Test{} | 	obj = protoexample.Test{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| 	req.Header.Add("Content-Type", MIMEPROTOBUF) | 	req.Header.Add("Content-Type", MIMEPROTOBUF) | ||||||
| 	err = ProtoBuf.Bind(req, &obj) | 	err = ProtoBuf.Bind(req, &obj) | ||||||
| @ -1032,14 +1263,14 @@ func testProtoBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func testMsgPackBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | func testMsgPackBodyBinding(t *testing.T, b Binding, name, path, badPath, body, badBody string) { | ||||||
| 	assert.Equal(t, b.Name(), name) | 	assert.Equal(t, name, b.Name()) | ||||||
| 
 | 
 | ||||||
| 	obj := FooStruct{} | 	obj := FooStruct{} | ||||||
| 	req := requestWithBody("POST", path, body) | 	req := requestWithBody("POST", path, body) | ||||||
| 	req.Header.Add("Content-Type", MIMEMSGPACK) | 	req.Header.Add("Content-Type", MIMEMSGPACK) | ||||||
| 	err := b.Bind(req, &obj) | 	err := b.Bind(req, &obj) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	assert.Equal(t, obj.Foo, "bar") | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 
 | 
 | ||||||
| 	obj = FooStruct{} | 	obj = FooStruct{} | ||||||
| 	req = requestWithBody("POST", badPath, badBody) | 	req = requestWithBody("POST", badPath, badBody) | ||||||
| @ -1052,3 +1283,12 @@ func requestWithBody(method, path, body string) (req *http.Request) { | |||||||
| 	req, _ = http.NewRequest(method, path, bytes.NewBufferString(body)) | 	req, _ = http.NewRequest(method, path, bytes.NewBufferString(body)) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func TestCanSet(t *testing.T) { | ||||||
|  | 	type CanSetStruct struct { | ||||||
|  | 		lowerStart string `form:"lower"` | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var c CanSetStruct | ||||||
|  | 	assert.Nil(t, mapForm(&c, nil)) | ||||||
|  | } | ||||||
|  | |||||||
| @ -18,19 +18,29 @@ type defaultValidator struct { | |||||||
| 
 | 
 | ||||||
| var _ StructValidator = &defaultValidator{} | var _ StructValidator = &defaultValidator{} | ||||||
| 
 | 
 | ||||||
|  | // ValidateStruct receives any kind of type, but only performed struct or pointer to struct type. | ||||||
| func (v *defaultValidator) ValidateStruct(obj interface{}) error { | func (v *defaultValidator) ValidateStruct(obj interface{}) error { | ||||||
| 	if kindOfData(obj) == reflect.Struct { | 	value := reflect.ValueOf(obj) | ||||||
|  | 	valueType := value.Kind() | ||||||
|  | 	if valueType == reflect.Ptr { | ||||||
|  | 		valueType = value.Elem().Kind() | ||||||
|  | 	} | ||||||
|  | 	if valueType == reflect.Struct { | ||||||
| 		v.lazyinit() | 		v.lazyinit() | ||||||
| 		if err := v.validate.Struct(obj); err != nil { | 		if err := v.validate.Struct(obj); err != nil { | ||||||
| 			return error(err) | 			return err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (v *defaultValidator) RegisterValidation(key string, fn validator.Func) error { | // Engine returns the underlying validator engine which powers the default | ||||||
|  | // Validator instance. This is useful if you want to register custom validations | ||||||
|  | // or struct level validations. See validator GoDoc for more info - | ||||||
|  | // https://godoc.org/gopkg.in/go-playground/validator.v8 | ||||||
|  | func (v *defaultValidator) Engine() interface{} { | ||||||
| 	v.lazyinit() | 	v.lazyinit() | ||||||
| 	return v.validate.RegisterValidation(key, fn) | 	return v.validate | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (v *defaultValidator) lazyinit() { | func (v *defaultValidator) lazyinit() { | ||||||
| @ -39,12 +49,3 @@ func (v *defaultValidator) lazyinit() { | |||||||
| 		v.validate = validator.New(config) | 		v.validate = validator.New(config) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 |  | ||||||
| func kindOfData(data interface{}) reflect.Kind { |  | ||||||
| 	value := reflect.ValueOf(data) |  | ||||||
| 	valueType := value.Kind() |  | ||||||
| 	if valueType == reflect.Ptr { |  | ||||||
| 		valueType = value.Elem().Kind() |  | ||||||
| 	} |  | ||||||
| 	return valueType |  | ||||||
| } |  | ||||||
|  | |||||||
| @ -20,7 +20,11 @@ func (formBinding) Bind(req *http.Request, obj interface{}) error { | |||||||
| 	if err := req.ParseForm(); err != nil { | 	if err := req.ParseForm(); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	req.ParseMultipartForm(defaultMemory) | 	if err := req.ParseMultipartForm(defaultMemory); err != nil { | ||||||
|  | 		if err != http.ErrNotMultipart { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	if err := mapForm(obj, req.Form); err != nil { | 	if err := mapForm(obj, req.Form); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -8,10 +8,19 @@ import ( | |||||||
| 	"errors" | 	"errors" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 	"strconv" | 	"strconv" | ||||||
|  | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | func mapUri(ptr interface{}, m map[string][]string) error { | ||||||
|  | 	return mapFormByTag(ptr, m, "uri") | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func mapForm(ptr interface{}, form map[string][]string) error { | func mapForm(ptr interface{}, form map[string][]string) error { | ||||||
|  | 	return mapFormByTag(ptr, form, "form") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func mapFormByTag(ptr interface{}, form map[string][]string, tag string) error { | ||||||
| 	typ := reflect.TypeOf(ptr).Elem() | 	typ := reflect.TypeOf(ptr).Elem() | ||||||
| 	val := reflect.ValueOf(ptr).Elem() | 	val := reflect.ValueOf(ptr).Elem() | ||||||
| 	for i := 0; i < typ.NumField(); i++ { | 	for i := 0; i < typ.NumField(); i++ { | ||||||
| @ -22,15 +31,31 @@ func mapForm(ptr interface{}, form map[string][]string) error { | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		structFieldKind := structField.Kind() | 		structFieldKind := structField.Kind() | ||||||
| 		inputFieldName := typeField.Tag.Get("form") | 		inputFieldName := typeField.Tag.Get(tag) | ||||||
|  | 		inputFieldNameList := strings.Split(inputFieldName, ",") | ||||||
|  | 		inputFieldName = inputFieldNameList[0] | ||||||
|  | 		var defaultValue string | ||||||
|  | 		if len(inputFieldNameList) > 1 { | ||||||
|  | 			defaultList := strings.SplitN(inputFieldNameList[1], "=", 2) | ||||||
|  | 			if defaultList[0] == "default" { | ||||||
|  | 				defaultValue = defaultList[1] | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
| 		if inputFieldName == "" { | 		if inputFieldName == "" { | ||||||
| 			inputFieldName = typeField.Name | 			inputFieldName = typeField.Name | ||||||
| 
 | 
 | ||||||
| 			// if "form" tag is nil, we inspect if the field is a struct. | 			// if "form" tag is nil, we inspect if the field is a struct or struct pointer. | ||||||
| 			// this would not make sense for JSON parsing but it does for a form | 			// this would not make sense for JSON parsing but it does for a form | ||||||
| 			// since data is flatten | 			// since data is flatten | ||||||
|  | 			if structFieldKind == reflect.Ptr { | ||||||
|  | 				if !structField.Elem().IsValid() { | ||||||
|  | 					structField.Set(reflect.New(structField.Type().Elem())) | ||||||
|  | 				} | ||||||
|  | 				structField = structField.Elem() | ||||||
|  | 				structFieldKind = structField.Kind() | ||||||
|  | 			} | ||||||
| 			if structFieldKind == reflect.Struct { | 			if structFieldKind == reflect.Struct { | ||||||
| 				err := mapForm(structField.Addr().Interface(), form) | 				err := mapFormByTag(structField.Addr().Interface(), form, tag) | ||||||
| 				if err != nil { | 				if err != nil { | ||||||
| 					return err | 					return err | ||||||
| 				} | 				} | ||||||
| @ -38,8 +63,13 @@ func mapForm(ptr interface{}, form map[string][]string) error { | |||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 		inputValue, exists := form[inputFieldName] | 		inputValue, exists := form[inputFieldName] | ||||||
|  | 
 | ||||||
| 		if !exists { | 		if !exists { | ||||||
| 			continue | 			if defaultValue == "" { | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			inputValue = make([]string, 1) | ||||||
|  | 			inputValue[0] = defaultValue | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		numElems := len(inputValue) | 		numElems := len(inputValue) | ||||||
| @ -52,16 +82,16 @@ func mapForm(ptr interface{}, form map[string][]string) error { | |||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			val.Field(i).Set(slice) | 			val.Field(i).Set(slice) | ||||||
| 		} else { | 			continue | ||||||
| 			if _, isTime := structField.Interface().(time.Time); isTime { | 		} | ||||||
| 				if err := setTimeField(inputValue[0], typeField, structField); err != nil { | 		if _, isTime := structField.Interface().(time.Time); isTime { | ||||||
| 					return err | 			if err := setTimeField(inputValue[0], typeField, structField); err != nil { | ||||||
| 				} |  | ||||||
| 				continue |  | ||||||
| 			} |  | ||||||
| 			if err := setWithProperType(typeField.Type.Kind(), inputValue[0], structField); err != nil { |  | ||||||
| 				return err | 				return err | ||||||
| 			} | 			} | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if err := setWithProperType(typeField.Type.Kind(), inputValue[0], structField); err != nil { | ||||||
|  | 			return err | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return nil | ||||||
| @ -97,6 +127,12 @@ func setWithProperType(valueKind reflect.Kind, val string, structField reflect.V | |||||||
| 		return setFloatField(val, 64, structField) | 		return setFloatField(val, 64, structField) | ||||||
| 	case reflect.String: | 	case reflect.String: | ||||||
| 		structField.SetString(val) | 		structField.SetString(val) | ||||||
|  | 	case reflect.Ptr: | ||||||
|  | 		if !structField.Elem().IsValid() { | ||||||
|  | 			structField.Set(reflect.New(structField.Type().Elem())) | ||||||
|  | 		} | ||||||
|  | 		structFieldElem := structField.Elem() | ||||||
|  | 		return setWithProperType(structFieldElem.Kind(), val, structFieldElem) | ||||||
| 	default: | 	default: | ||||||
| 		return errors.New("Unknown type") | 		return errors.New("Unknown type") | ||||||
| 	} | 	} | ||||||
| @ -133,7 +169,7 @@ func setBoolField(val string, field reflect.Value) error { | |||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		field.SetBool(boolVal) | 		field.SetBool(boolVal) | ||||||
| 	} | 	} | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func setFloatField(val string, bitSize int, field reflect.Value) error { | func setFloatField(val string, bitSize int, field reflect.Value) error { | ||||||
| @ -150,7 +186,7 @@ func setFloatField(val string, bitSize int, field reflect.Value) error { | |||||||
| func setTimeField(val string, structField reflect.StructField, value reflect.Value) error { | func setTimeField(val string, structField reflect.StructField, value reflect.Value) error { | ||||||
| 	timeFormat := structField.Tag.Get("time_format") | 	timeFormat := structField.Tag.Get("time_format") | ||||||
| 	if timeFormat == "" { | 	if timeFormat == "" { | ||||||
| 		return errors.New("Blank time format") | 		timeFormat = time.RFC3339 | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if val == "" { | 	if val == "" { | ||||||
|  | |||||||
| @ -5,11 +5,17 @@ | |||||||
| package binding | package binding | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin/json" | 	"github.com/gin-gonic/gin/internal/json" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // EnableDecoderUseNumber is used to call the UseNumber method on the JSON | ||||||
|  | // Decoder instance. UseNumber causes the Decoder to unmarshal a number into an | ||||||
|  | // interface{} as a Number instead of as a float64. | ||||||
| var EnableDecoderUseNumber = false | var EnableDecoderUseNumber = false | ||||||
| 
 | 
 | ||||||
| type jsonBinding struct{} | type jsonBinding struct{} | ||||||
| @ -19,7 +25,18 @@ func (jsonBinding) Name() string { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (jsonBinding) Bind(req *http.Request, obj interface{}) error { | func (jsonBinding) Bind(req *http.Request, obj interface{}) error { | ||||||
| 	decoder := json.NewDecoder(req.Body) | 	if req == nil || req.Body == nil { | ||||||
|  | 		return fmt.Errorf("invalid request") | ||||||
|  | 	} | ||||||
|  | 	return decodeJSON(req.Body, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (jsonBinding) BindBody(body []byte, obj interface{}) error { | ||||||
|  | 	return decodeJSON(bytes.NewReader(body), obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func decodeJSON(r io.Reader, obj interface{}) error { | ||||||
|  | 	decoder := json.NewDecoder(r) | ||||||
| 	if EnableDecoderUseNumber { | 	if EnableDecoderUseNumber { | ||||||
| 		decoder.UseNumber() | 		decoder.UseNumber() | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -5,6 +5,8 @@ | |||||||
| package binding | package binding | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 
 | 
 | ||||||
| 	"github.com/ugorji/go/codec" | 	"github.com/ugorji/go/codec" | ||||||
| @ -17,7 +19,16 @@ func (msgpackBinding) Name() string { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (msgpackBinding) Bind(req *http.Request, obj interface{}) error { | func (msgpackBinding) Bind(req *http.Request, obj interface{}) error { | ||||||
| 	if err := codec.NewDecoder(req.Body, new(codec.MsgpackHandle)).Decode(&obj); err != nil { | 	return decodeMsgPack(req.Body, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (msgpackBinding) BindBody(body []byte, obj interface{}) error { | ||||||
|  | 	return decodeMsgPack(bytes.NewReader(body), obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func decodeMsgPack(r io.Reader, obj interface{}) error { | ||||||
|  | 	cdc := new(codec.MsgpackHandle) | ||||||
|  | 	if err := codec.NewDecoder(r, cdc).Decode(&obj); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	return validate(obj) | 	return validate(obj) | ||||||
|  | |||||||
| @ -17,19 +17,20 @@ func (protobufBinding) Name() string { | |||||||
| 	return "protobuf" | 	return "protobuf" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (protobufBinding) Bind(req *http.Request, obj interface{}) error { | func (b protobufBinding) Bind(req *http.Request, obj interface{}) error { | ||||||
| 
 |  | ||||||
| 	buf, err := ioutil.ReadAll(req.Body) | 	buf, err := ioutil.ReadAll(req.Body) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | 	return b.BindBody(buf, obj) | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| 	if err = proto.Unmarshal(buf, obj.(proto.Message)); err != nil { | func (protobufBinding) BindBody(body []byte, obj interface{}) error { | ||||||
|  | 	if err := proto.Unmarshal(body, obj.(proto.Message)); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 
 | 	// Here it's same to return validate(obj), but util now we can't add | ||||||
| 	//Here it's same to return validate(obj), but util now we cann't add `binding:""` to the struct | 	// `binding:""` to the struct which automatically generate by gen-proto | ||||||
| 	//which automatically generate by gen-proto |  | ||||||
| 	return nil | 	return nil | ||||||
| 	//return validate(obj) | 	// return validate(obj) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										18
									
								
								binding/uri.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								binding/uri.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,18 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package binding | ||||||
|  | 
 | ||||||
|  | type uriBinding struct{} | ||||||
|  | 
 | ||||||
|  | func (uriBinding) Name() string { | ||||||
|  | 	return "uri" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (uriBinding) BindUri(m map[string][]string, obj interface{}) error { | ||||||
|  | 	if err := mapUri(obj, m); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return validate(obj) | ||||||
|  | } | ||||||
| @ -176,7 +176,7 @@ func TestValidatePrimitives(t *testing.T) { | |||||||
| 	obj := Object{"foo": "bar", "bar": 1} | 	obj := Object{"foo": "bar", "bar": 1} | ||||||
| 	assert.NoError(t, validate(obj)) | 	assert.NoError(t, validate(obj)) | ||||||
| 	assert.NoError(t, validate(&obj)) | 	assert.NoError(t, validate(&obj)) | ||||||
| 	assert.Equal(t, obj, Object{"foo": "bar", "bar": 1}) | 	assert.Equal(t, Object{"foo": "bar", "bar": 1}, obj) | ||||||
| 
 | 
 | ||||||
| 	obj2 := []Object{{"foo": "bar", "bar": 1}, {"foo": "bar", "bar": 1}} | 	obj2 := []Object{{"foo": "bar", "bar": 1}, {"foo": "bar", "bar": 1}} | ||||||
| 	assert.NoError(t, validate(obj2)) | 	assert.NoError(t, validate(obj2)) | ||||||
| @ -185,12 +185,12 @@ func TestValidatePrimitives(t *testing.T) { | |||||||
| 	nu := 10 | 	nu := 10 | ||||||
| 	assert.NoError(t, validate(nu)) | 	assert.NoError(t, validate(nu)) | ||||||
| 	assert.NoError(t, validate(&nu)) | 	assert.NoError(t, validate(&nu)) | ||||||
| 	assert.Equal(t, nu, 10) | 	assert.Equal(t, 10, nu) | ||||||
| 
 | 
 | ||||||
| 	str := "value" | 	str := "value" | ||||||
| 	assert.NoError(t, validate(str)) | 	assert.NoError(t, validate(str)) | ||||||
| 	assert.NoError(t, validate(&str)) | 	assert.NoError(t, validate(&str)) | ||||||
| 	assert.Equal(t, str, "value") | 	assert.Equal(t, "value", str) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // structCustomValidation is a helper struct we use to check that | // structCustomValidation is a helper struct we use to check that | ||||||
| @ -214,11 +214,14 @@ func notOne( | |||||||
| 	return false | 	return false | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestRegisterValidation(t *testing.T) { | func TestValidatorEngine(t *testing.T) { | ||||||
| 	// This validates that the function `notOne` matches | 	// This validates that the function `notOne` matches | ||||||
| 	// the expected function signature by `defaultValidator` | 	// the expected function signature by `defaultValidator` | ||||||
| 	// and by extension the validator library. | 	// and by extension the validator library. | ||||||
| 	err := Validator.RegisterValidation("notone", notOne) | 	engine, ok := Validator.Engine().(*validator.Validate) | ||||||
|  | 	assert.True(t, ok) | ||||||
|  | 
 | ||||||
|  | 	err := engine.RegisterValidation("notone", notOne) | ||||||
| 	// Check that we can register custom validation without error | 	// Check that we can register custom validation without error | ||||||
| 	assert.Nil(t, err) | 	assert.Nil(t, err) | ||||||
| 
 | 
 | ||||||
| @ -228,6 +231,6 @@ func TestRegisterValidation(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	// Check that we got back non-nil errs | 	// Check that we got back non-nil errs | ||||||
| 	assert.NotNil(t, errs) | 	assert.NotNil(t, errs) | ||||||
| 	// Check that the error matches expactation | 	// Check that the error matches expectation | ||||||
| 	assert.Error(t, errs, "", "", "notone") | 	assert.Error(t, errs, "", "", "notone") | ||||||
| } | } | ||||||
|  | |||||||
| @ -5,7 +5,9 @@ | |||||||
| package binding | package binding | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"bytes" | ||||||
| 	"encoding/xml" | 	"encoding/xml" | ||||||
|  | 	"io" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -16,7 +18,14 @@ func (xmlBinding) Name() string { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (xmlBinding) Bind(req *http.Request, obj interface{}) error { | func (xmlBinding) Bind(req *http.Request, obj interface{}) error { | ||||||
| 	decoder := xml.NewDecoder(req.Body) | 	return decodeXML(req.Body, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (xmlBinding) BindBody(body []byte, obj interface{}) error { | ||||||
|  | 	return decodeXML(bytes.NewReader(body), obj) | ||||||
|  | } | ||||||
|  | func decodeXML(r io.Reader, obj interface{}) error { | ||||||
|  | 	decoder := xml.NewDecoder(r) | ||||||
| 	if err := decoder.Decode(obj); err != nil { | 	if err := decoder.Decode(obj); err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
|  | |||||||
							
								
								
									
										35
									
								
								binding/yaml.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								binding/yaml.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,35 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package binding | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"bytes" | ||||||
|  | 	"io" | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
|  | 	"gopkg.in/yaml.v2" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | type yamlBinding struct{} | ||||||
|  | 
 | ||||||
|  | func (yamlBinding) Name() string { | ||||||
|  | 	return "yaml" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (yamlBinding) Bind(req *http.Request, obj interface{}) error { | ||||||
|  | 	return decodeYAML(req.Body, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (yamlBinding) BindBody(body []byte, obj interface{}) error { | ||||||
|  | 	return decodeYAML(bytes.NewReader(body), obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func decodeYAML(r io.Reader, obj interface{}) error { | ||||||
|  | 	decoder := yaml.NewDecoder(r) | ||||||
|  | 	if err := decoder.Decode(obj); err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return validate(obj) | ||||||
|  | } | ||||||
							
								
								
									
										234
									
								
								context.go
									
									
									
									
									
								
							
							
						
						
									
										234
									
								
								context.go
									
									
									
									
									
								
							| @ -31,6 +31,8 @@ const ( | |||||||
| 	MIMEPlain             = binding.MIMEPlain | 	MIMEPlain             = binding.MIMEPlain | ||||||
| 	MIMEPOSTForm          = binding.MIMEPOSTForm | 	MIMEPOSTForm          = binding.MIMEPOSTForm | ||||||
| 	MIMEMultipartPOSTForm = binding.MIMEMultipartPOSTForm | 	MIMEMultipartPOSTForm = binding.MIMEMultipartPOSTForm | ||||||
|  | 	MIMEYAML              = binding.MIMEYAML | ||||||
|  | 	BodyBytesKey          = "_gin-gonic/gin/bodybyteskey" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const abortIndex int8 = math.MaxInt8 / 2 | const abortIndex int8 = math.MaxInt8 / 2 | ||||||
| @ -103,8 +105,9 @@ func (c *Context) Handler() HandlerFunc { | |||||||
| // See example in GitHub. | // See example in GitHub. | ||||||
| func (c *Context) Next() { | func (c *Context) Next() { | ||||||
| 	c.index++ | 	c.index++ | ||||||
| 	for s := int8(len(c.handlers)); c.index < s; c.index++ { | 	for c.index < int8(len(c.handlers)) { | ||||||
| 		c.handlers[c.index](c) | 		c.handlers[c.index](c) | ||||||
|  | 		c.index++ | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -158,16 +161,15 @@ func (c *Context) Error(err error) *Error { | |||||||
| 	if err == nil { | 	if err == nil { | ||||||
| 		panic("err is nil") | 		panic("err is nil") | ||||||
| 	} | 	} | ||||||
| 	var parsedError *Error | 
 | ||||||
| 	switch err.(type) { | 	parsedError, ok := err.(*Error) | ||||||
| 	case *Error: | 	if !ok { | ||||||
| 		parsedError = err.(*Error) |  | ||||||
| 	default: |  | ||||||
| 		parsedError = &Error{ | 		parsedError = &Error{ | ||||||
| 			Err:  err, | 			Err:  err, | ||||||
| 			Type: ErrorTypePrivate, | 			Type: ErrorTypePrivate, | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
| 	c.Errors = append(c.Errors, parsedError) | 	c.Errors = append(c.Errors, parsedError) | ||||||
| 	return parsedError | 	return parsedError | ||||||
| } | } | ||||||
| @ -360,6 +362,18 @@ func (c *Context) GetQueryArray(key string) ([]string, bool) { | |||||||
| 	return []string{}, false | 	return []string{}, false | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // QueryMap returns a map for a given query key. | ||||||
|  | func (c *Context) QueryMap(key string) map[string]string { | ||||||
|  | 	dicts, _ := c.GetQueryMap(key) | ||||||
|  | 	return dicts | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GetQueryMap returns a map for a given query key, plus a boolean value | ||||||
|  | // whether at least one value exists for the given key. | ||||||
|  | func (c *Context) GetQueryMap(key string) (map[string]string, bool) { | ||||||
|  | 	return c.get(c.Request.URL.Query(), key) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // PostForm returns the specified key from a POST urlencoded form or multipart form | // PostForm returns the specified key from a POST urlencoded form or multipart form | ||||||
| // when it exists, otherwise it returns an empty string `("")`. | // when it exists, otherwise it returns an empty string `("")`. | ||||||
| func (c *Context) PostForm(key string) string { | func (c *Context) PostForm(key string) string { | ||||||
| @ -402,8 +416,11 @@ func (c *Context) PostFormArray(key string) []string { | |||||||
| // a boolean value whether at least one value exists for the given key. | // a boolean value whether at least one value exists for the given key. | ||||||
| func (c *Context) GetPostFormArray(key string) ([]string, bool) { | func (c *Context) GetPostFormArray(key string) ([]string, bool) { | ||||||
| 	req := c.Request | 	req := c.Request | ||||||
| 	req.ParseForm() | 	if err := req.ParseMultipartForm(c.engine.MaxMultipartMemory); err != nil { | ||||||
| 	req.ParseMultipartForm(c.engine.MaxMultipartMemory) | 		if err != http.ErrNotMultipart { | ||||||
|  | 			debugPrint("error on parse multipart form array: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	if values := req.PostForm[key]; len(values) > 0 { | 	if values := req.PostForm[key]; len(values) > 0 { | ||||||
| 		return values, true | 		return values, true | ||||||
| 	} | 	} | ||||||
| @ -415,8 +432,52 @@ func (c *Context) GetPostFormArray(key string) ([]string, bool) { | |||||||
| 	return []string{}, false | 	return []string{}, false | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // PostFormMap returns a map for a given form key. | ||||||
|  | func (c *Context) PostFormMap(key string) map[string]string { | ||||||
|  | 	dicts, _ := c.GetPostFormMap(key) | ||||||
|  | 	return dicts | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // GetPostFormMap returns a map for a given form key, plus a boolean value | ||||||
|  | // whether at least one value exists for the given key. | ||||||
|  | func (c *Context) GetPostFormMap(key string) (map[string]string, bool) { | ||||||
|  | 	req := c.Request | ||||||
|  | 	if err := req.ParseMultipartForm(c.engine.MaxMultipartMemory); err != nil { | ||||||
|  | 		if err != http.ErrNotMultipart { | ||||||
|  | 			debugPrint("error on parse multipart form map: %v", err) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	dicts, exist := c.get(req.PostForm, key) | ||||||
|  | 
 | ||||||
|  | 	if !exist && req.MultipartForm != nil && req.MultipartForm.File != nil { | ||||||
|  | 		dicts, exist = c.get(req.MultipartForm.Value, key) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return dicts, exist | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // get is an internal method and returns a map which satisfy conditions. | ||||||
|  | func (c *Context) get(m map[string][]string, key string) (map[string]string, bool) { | ||||||
|  | 	dicts := make(map[string]string) | ||||||
|  | 	exist := false | ||||||
|  | 	for k, v := range m { | ||||||
|  | 		if i := strings.IndexByte(k, '['); i >= 1 && k[0:i] == key { | ||||||
|  | 			if j := strings.IndexByte(k[i+1:], ']'); j >= 1 { | ||||||
|  | 				exist = true | ||||||
|  | 				dicts[k[i+1:][:j]] = v[0] | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return dicts, exist | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // FormFile returns the first file for the provided form key. | // FormFile returns the first file for the provided form key. | ||||||
| func (c *Context) FormFile(name string) (*multipart.FileHeader, error) { | func (c *Context) FormFile(name string) (*multipart.FileHeader, error) { | ||||||
|  | 	if c.Request.MultipartForm == nil { | ||||||
|  | 		if err := c.Request.ParseMultipartForm(c.engine.MaxMultipartMemory); err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
| 	_, fh, err := c.Request.FormFile(name) | 	_, fh, err := c.Request.FormFile(name) | ||||||
| 	return fh, err | 	return fh, err | ||||||
| } | } | ||||||
| @ -441,8 +502,8 @@ func (c *Context) SaveUploadedFile(file *multipart.FileHeader, dst string) error | |||||||
| 	} | 	} | ||||||
| 	defer out.Close() | 	defer out.Close() | ||||||
| 
 | 
 | ||||||
| 	io.Copy(out, src) | 	_, err = io.Copy(out, src) | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Bind checks the Content-Type to select a binding engine automatically, | // Bind checks the Content-Type to select a binding engine automatically, | ||||||
| @ -463,20 +524,40 @@ func (c *Context) BindJSON(obj interface{}) error { | |||||||
| 	return c.MustBindWith(obj, binding.JSON) | 	return c.MustBindWith(obj, binding.JSON) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // BindXML is a shortcut for c.MustBindWith(obj, binding.BindXML). | ||||||
|  | func (c *Context) BindXML(obj interface{}) error { | ||||||
|  | 	return c.MustBindWith(obj, binding.XML) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // BindQuery is a shortcut for c.MustBindWith(obj, binding.Query). | // BindQuery is a shortcut for c.MustBindWith(obj, binding.Query). | ||||||
| func (c *Context) BindQuery(obj interface{}) error { | func (c *Context) BindQuery(obj interface{}) error { | ||||||
| 	return c.MustBindWith(obj, binding.Query) | 	return c.MustBindWith(obj, binding.Query) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // MustBindWith binds the passed struct pointer using the specified binding engine. | // BindYAML is a shortcut for c.MustBindWith(obj, binding.YAML). | ||||||
| // It will abort the request with HTTP 400 if any error ocurrs. | func (c *Context) BindYAML(obj interface{}) error { | ||||||
| // See the binding package. | 	return c.MustBindWith(obj, binding.YAML) | ||||||
| func (c *Context) MustBindWith(obj interface{}, b binding.Binding) (err error) { | } | ||||||
| 	if err = c.ShouldBindWith(obj, b); err != nil { |  | ||||||
| 		c.AbortWithError(400, err).SetType(ErrorTypeBind) |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	return | // BindUri binds the passed struct pointer using binding.Uri. | ||||||
|  | // It will abort the request with HTTP 400 if any error occurs. | ||||||
|  | func (c *Context) BindUri(obj interface{}) error { | ||||||
|  | 	if err := c.ShouldBindUri(obj); err != nil { | ||||||
|  | 		c.AbortWithError(http.StatusBadRequest, err).SetType(ErrorTypeBind) // nolint: errcheck | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // MustBindWith binds the passed struct pointer using the specified binding engine. | ||||||
|  | // It will abort the request with HTTP 400 if any error occurs. | ||||||
|  | // See the binding package. | ||||||
|  | func (c *Context) MustBindWith(obj interface{}, b binding.Binding) error { | ||||||
|  | 	if err := c.ShouldBindWith(obj, b); err != nil { | ||||||
|  | 		c.AbortWithError(http.StatusBadRequest, err).SetType(ErrorTypeBind) // nolint: errcheck | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // ShouldBind checks the Content-Type to select a binding engine automatically, | // ShouldBind checks the Content-Type to select a binding engine automatically, | ||||||
| @ -497,31 +578,68 @@ func (c *Context) ShouldBindJSON(obj interface{}) error { | |||||||
| 	return c.ShouldBindWith(obj, binding.JSON) | 	return c.ShouldBindWith(obj, binding.JSON) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ShouldBindXML is a shortcut for c.ShouldBindWith(obj, binding.XML). | ||||||
|  | func (c *Context) ShouldBindXML(obj interface{}) error { | ||||||
|  | 	return c.ShouldBindWith(obj, binding.XML) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // ShouldBindQuery is a shortcut for c.ShouldBindWith(obj, binding.Query). | // ShouldBindQuery is a shortcut for c.ShouldBindWith(obj, binding.Query). | ||||||
| func (c *Context) ShouldBindQuery(obj interface{}) error { | func (c *Context) ShouldBindQuery(obj interface{}) error { | ||||||
| 	return c.ShouldBindWith(obj, binding.Query) | 	return c.ShouldBindWith(obj, binding.Query) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ShouldBindYAML is a shortcut for c.ShouldBindWith(obj, binding.YAML). | ||||||
|  | func (c *Context) ShouldBindYAML(obj interface{}) error { | ||||||
|  | 	return c.ShouldBindWith(obj, binding.YAML) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // ShouldBindUri binds the passed struct pointer using the specified binding engine. | ||||||
|  | func (c *Context) ShouldBindUri(obj interface{}) error { | ||||||
|  | 	m := make(map[string][]string) | ||||||
|  | 	for _, v := range c.Params { | ||||||
|  | 		m[v.Key] = []string{v.Value} | ||||||
|  | 	} | ||||||
|  | 	return binding.Uri.BindUri(m, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // ShouldBindWith binds the passed struct pointer using the specified binding engine. | // ShouldBindWith binds the passed struct pointer using the specified binding engine. | ||||||
| // See the binding package. | // See the binding package. | ||||||
| func (c *Context) ShouldBindWith(obj interface{}, b binding.Binding) error { | func (c *Context) ShouldBindWith(obj interface{}, b binding.Binding) error { | ||||||
| 	return b.Bind(c.Request, obj) | 	return b.Bind(c.Request, obj) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ShouldBindBodyWith is similar with ShouldBindWith, but it stores the request | ||||||
|  | // body into the context, and reuse when it is called again. | ||||||
|  | // | ||||||
|  | // NOTE: This method reads the body before binding. So you should use | ||||||
|  | // ShouldBindWith for better performance if you need to call only once. | ||||||
|  | func (c *Context) ShouldBindBodyWith(obj interface{}, bb binding.BindingBody) (err error) { | ||||||
|  | 	var body []byte | ||||||
|  | 	if cb, ok := c.Get(BodyBytesKey); ok { | ||||||
|  | 		if cbb, ok := cb.([]byte); ok { | ||||||
|  | 			body = cbb | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	if body == nil { | ||||||
|  | 		body, err = ioutil.ReadAll(c.Request.Body) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
|  | 		c.Set(BodyBytesKey, body) | ||||||
|  | 	} | ||||||
|  | 	return bb.BindBody(body, obj) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // ClientIP implements a best effort algorithm to return the real client IP, it parses | // ClientIP implements a best effort algorithm to return the real client IP, it parses | ||||||
| // X-Real-IP and X-Forwarded-For in order to work properly with reverse-proxies such us: nginx or haproxy. | // X-Real-IP and X-Forwarded-For in order to work properly with reverse-proxies such us: nginx or haproxy. | ||||||
| // Use X-Forwarded-For before X-Real-Ip as nginx uses X-Real-Ip with the proxy's IP. | // Use X-Forwarded-For before X-Real-Ip as nginx uses X-Real-Ip with the proxy's IP. | ||||||
| func (c *Context) ClientIP() string { | func (c *Context) ClientIP() string { | ||||||
| 	if c.engine.ForwardedByClientIP { | 	if c.engine.ForwardedByClientIP { | ||||||
| 		clientIP := c.requestHeader("X-Forwarded-For") | 		clientIP := c.requestHeader("X-Forwarded-For") | ||||||
| 		if index := strings.IndexByte(clientIP, ','); index >= 0 { | 		clientIP = strings.TrimSpace(strings.Split(clientIP, ",")[0]) | ||||||
| 			clientIP = clientIP[0:index] | 		if clientIP == "" { | ||||||
|  | 			clientIP = strings.TrimSpace(c.requestHeader("X-Real-Ip")) | ||||||
| 		} | 		} | ||||||
| 		clientIP = strings.TrimSpace(clientIP) |  | ||||||
| 		if clientIP != "" { |  | ||||||
| 			return clientIP |  | ||||||
| 		} |  | ||||||
| 		clientIP = strings.TrimSpace(c.requestHeader("X-Real-Ip")) |  | ||||||
| 		if clientIP != "" { | 		if clientIP != "" { | ||||||
| 			return clientIP | 			return clientIP | ||||||
| 		} | 		} | ||||||
| @ -568,9 +686,9 @@ func bodyAllowedForStatus(status int) bool { | |||||||
| 	switch { | 	switch { | ||||||
| 	case status >= 100 && status <= 199: | 	case status >= 100 && status <= 199: | ||||||
| 		return false | 		return false | ||||||
| 	case status == 204: | 	case status == http.StatusNoContent: | ||||||
| 		return false | 		return false | ||||||
| 	case status == 304: | 	case status == http.StatusNotModified: | ||||||
| 		return false | 		return false | ||||||
| 	} | 	} | ||||||
| 	return true | 	return true | ||||||
| @ -587,9 +705,9 @@ func (c *Context) Status(code int) { | |||||||
| func (c *Context) Header(key, value string) { | func (c *Context) Header(key, value string) { | ||||||
| 	if value == "" { | 	if value == "" { | ||||||
| 		c.Writer.Header().Del(key) | 		c.Writer.Header().Del(key) | ||||||
| 	} else { | 		return | ||||||
| 		c.Writer.Header().Set(key, value) |  | ||||||
| 	} | 	} | ||||||
|  | 	c.Writer.Header().Set(key, value) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // GetHeader returns value from request headers. | // GetHeader returns value from request headers. | ||||||
| @ -633,6 +751,7 @@ func (c *Context) Cookie(name string) (string, error) { | |||||||
| 	return val, nil | 	return val, nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render writes the response headers and calls render.Render to render data. | ||||||
| func (c *Context) Render(code int, r render.Render) { | func (c *Context) Render(code int, r render.Render) { | ||||||
| 	c.Status(code) | 	c.Status(code) | ||||||
| 
 | 
 | ||||||
| @ -670,12 +789,30 @@ func (c *Context) SecureJSON(code int, obj interface{}) { | |||||||
| 	c.Render(code, render.SecureJSON{Prefix: c.engine.secureJsonPrefix, Data: obj}) | 	c.Render(code, render.SecureJSON{Prefix: c.engine.secureJsonPrefix, Data: obj}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // JSONP serializes the given struct as JSON into the response body. | ||||||
|  | // It add padding to response body to request data from a server residing in a different domain than the client. | ||||||
|  | // It also sets the Content-Type as "application/javascript". | ||||||
|  | func (c *Context) JSONP(code int, obj interface{}) { | ||||||
|  | 	callback := c.DefaultQuery("callback", "") | ||||||
|  | 	if callback == "" { | ||||||
|  | 		c.Render(code, render.JSON{Data: obj}) | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	c.Render(code, render.JsonpJSON{Callback: callback, Data: obj}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // JSON serializes the given struct as JSON into the response body. | // JSON serializes the given struct as JSON into the response body. | ||||||
| // It also sets the Content-Type as "application/json". | // It also sets the Content-Type as "application/json". | ||||||
| func (c *Context) JSON(code int, obj interface{}) { | func (c *Context) JSON(code int, obj interface{}) { | ||||||
| 	c.Render(code, render.JSON{Data: obj}) | 	c.Render(code, render.JSON{Data: obj}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // AsciiJSON serializes the given struct as JSON into the response body with unicode to ASCII string. | ||||||
|  | // It also sets the Content-Type as "application/json". | ||||||
|  | func (c *Context) AsciiJSON(code int, obj interface{}) { | ||||||
|  | 	c.Render(code, render.AsciiJSON{Data: obj}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // XML serializes the given struct as XML into the response body. | // XML serializes the given struct as XML into the response body. | ||||||
| // It also sets the Content-Type as "application/xml". | // It also sets the Content-Type as "application/xml". | ||||||
| func (c *Context) XML(code int, obj interface{}) { | func (c *Context) XML(code int, obj interface{}) { | ||||||
| @ -687,6 +824,11 @@ func (c *Context) YAML(code int, obj interface{}) { | |||||||
| 	c.Render(code, render.YAML{Data: obj}) | 	c.Render(code, render.YAML{Data: obj}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ProtoBuf serializes the given struct as ProtoBuf into the response body. | ||||||
|  | func (c *Context) ProtoBuf(code int, obj interface{}) { | ||||||
|  | 	c.Render(code, render.ProtoBuf{Data: obj}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // String writes the given string into the response body. | // String writes the given string into the response body. | ||||||
| func (c *Context) String(code int, format string, values ...interface{}) { | func (c *Context) String(code int, format string, values ...interface{}) { | ||||||
| 	c.Render(code, render.String{Format: format, Data: values}) | 	c.Render(code, render.String{Format: format, Data: values}) | ||||||
| @ -709,6 +851,16 @@ func (c *Context) Data(code int, contentType string, data []byte) { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // DataFromReader writes the specified reader into the body stream and updates the HTTP code. | ||||||
|  | func (c *Context) DataFromReader(code int, contentLength int64, contentType string, reader io.Reader, extraHeaders map[string]string) { | ||||||
|  | 	c.Render(code, render.Reader{ | ||||||
|  | 		Headers:       extraHeaders, | ||||||
|  | 		ContentType:   contentType, | ||||||
|  | 		ContentLength: contentLength, | ||||||
|  | 		Reader:        reader, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // File writes the specified file into the body stream in a efficient way. | // File writes the specified file into the body stream in a efficient way. | ||||||
| func (c *Context) File(filepath string) { | func (c *Context) File(filepath string) { | ||||||
| 	http.ServeFile(c.Writer, c.Request, filepath) | 	http.ServeFile(c.Writer, c.Request, filepath) | ||||||
| @ -722,7 +874,8 @@ func (c *Context) SSEvent(name string, message interface{}) { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Returns a boolean indicates "Is client disconnected in middle of stream" | // Stream sends a streaming response and returns a boolean | ||||||
|  | // indicates "Is client disconnected in middle of stream" | ||||||
| func (c *Context) Stream(step func(w io.Writer) bool) bool { | func (c *Context) Stream(step func(w io.Writer) bool) bool { | ||||||
| 	w := c.Writer | 	w := c.Writer | ||||||
| 	clientGone := w.CloseNotify() | 	clientGone := w.CloseNotify() | ||||||
| @ -744,6 +897,7 @@ func (c *Context) Stream(step func(w io.Writer) bool) bool { | |||||||
| /******** CONTENT NEGOTIATION *******/ | /******** CONTENT NEGOTIATION *******/ | ||||||
| /************************************/ | /************************************/ | ||||||
| 
 | 
 | ||||||
|  | // Negotiate contains all negotiations data. | ||||||
| type Negotiate struct { | type Negotiate struct { | ||||||
| 	Offered  []string | 	Offered  []string | ||||||
| 	HTMLName string | 	HTMLName string | ||||||
| @ -753,6 +907,7 @@ type Negotiate struct { | |||||||
| 	Data     interface{} | 	Data     interface{} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Negotiate calls different Render according acceptable Accept format. | ||||||
| func (c *Context) Negotiate(code int, config Negotiate) { | func (c *Context) Negotiate(code int, config Negotiate) { | ||||||
| 	switch c.NegotiateFormat(config.Offered...) { | 	switch c.NegotiateFormat(config.Offered...) { | ||||||
| 	case binding.MIMEJSON: | 	case binding.MIMEJSON: | ||||||
| @ -768,10 +923,11 @@ func (c *Context) Negotiate(code int, config Negotiate) { | |||||||
| 		c.XML(code, data) | 		c.XML(code, data) | ||||||
| 
 | 
 | ||||||
| 	default: | 	default: | ||||||
| 		c.AbortWithError(http.StatusNotAcceptable, errors.New("the accepted formats are not offered by the server")) | 		c.AbortWithError(http.StatusNotAcceptable, errors.New("the accepted formats are not offered by the server")) // nolint: errcheck | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // NegotiateFormat returns an acceptable Accept format. | ||||||
| func (c *Context) NegotiateFormat(offered ...string) string { | func (c *Context) NegotiateFormat(offered ...string) string { | ||||||
| 	assert1(len(offered) > 0, "you must provide at least one offer") | 	assert1(len(offered) > 0, "you must provide at least one offer") | ||||||
| 
 | 
 | ||||||
| @ -791,6 +947,7 @@ func (c *Context) NegotiateFormat(offered ...string) string { | |||||||
| 	return "" | 	return "" | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // SetAccepted sets Accept header data. | ||||||
| func (c *Context) SetAccepted(formats ...string) { | func (c *Context) SetAccepted(formats ...string) { | ||||||
| 	c.Accepted = formats | 	c.Accepted = formats | ||||||
| } | } | ||||||
| @ -799,18 +956,33 @@ func (c *Context) SetAccepted(formats ...string) { | |||||||
| /***** GOLANG.ORG/X/NET/CONTEXT *****/ | /***** GOLANG.ORG/X/NET/CONTEXT *****/ | ||||||
| /************************************/ | /************************************/ | ||||||
| 
 | 
 | ||||||
|  | // Deadline returns the time when work done on behalf of this context | ||||||
|  | // should be canceled. Deadline returns ok==false when no deadline is | ||||||
|  | // set. Successive calls to Deadline return the same results. | ||||||
| func (c *Context) Deadline() (deadline time.Time, ok bool) { | func (c *Context) Deadline() (deadline time.Time, ok bool) { | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Done returns a channel that's closed when work done on behalf of this | ||||||
|  | // context should be canceled. Done may return nil if this context can | ||||||
|  | // never be canceled. Successive calls to Done return the same value. | ||||||
| func (c *Context) Done() <-chan struct{} { | func (c *Context) Done() <-chan struct{} { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Err returns a non-nil error value after Done is closed, | ||||||
|  | // successive calls to Err return the same error. | ||||||
|  | // If Done is not yet closed, Err returns nil. | ||||||
|  | // If Done is closed, Err returns a non-nil error explaining why: | ||||||
|  | // Canceled if the context was canceled | ||||||
|  | // or DeadlineExceeded if the context's deadline passed. | ||||||
| func (c *Context) Err() error { | func (c *Context) Err() error { | ||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Value returns the value associated with this context for key, or nil | ||||||
|  | // if no value is associated with key. Successive calls to Value with | ||||||
|  | // the same key returns the same result. | ||||||
| func (c *Context) Value(key interface{}) interface{} { | func (c *Context) Value(key interface{}) interface{} { | ||||||
| 	if key == 0 { | 	if key == 0 { | ||||||
| 		return c.Request | 		return c.Request | ||||||
|  | |||||||
							
								
								
									
										17
									
								
								context_17.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								context_17.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build go1.7 | ||||||
|  | 
 | ||||||
|  | package gin | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"github.com/gin-gonic/gin/render" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // PureJSON serializes the given struct as JSON into the response body. | ||||||
|  | // PureJSON, unlike JSON, does not replace special html characters with their unicode entities. | ||||||
|  | func (c *Context) PureJSON(code int, obj interface{}) { | ||||||
|  | 	c.Render(code, render.PureJSON{Data: obj}) | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								context_17_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								context_17_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,27 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build go1.7 | ||||||
|  | 
 | ||||||
|  | package gin | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 	"net/http/httptest" | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // Tests that the response is serialized as JSON | ||||||
|  | // and Content-Type is set to application/json | ||||||
|  | // and special HTML characters are preserved | ||||||
|  | func TestContextRenderPureJSON(t *testing.T) { | ||||||
|  | 	w := httptest.NewRecorder() | ||||||
|  | 	c, _ := CreateTestContext(w) | ||||||
|  | 	c.PureJSON(http.StatusCreated, H{"foo": "bar", "html": "<b>"}) | ||||||
|  | 	assert.Equal(t, http.StatusCreated, w.Code) | ||||||
|  | 	assert.Equal(t, "{\"foo\":\"bar\",\"html\":\"<b>\"}\n", w.Body.String()) | ||||||
|  | 	assert.Equal(t, "application/json; charset=utf-8", w.Header().Get("Content-Type")) | ||||||
|  | } | ||||||
							
								
								
									
										610
									
								
								context_test.go
									
									
									
									
									
								
							
							
						
						
									
										610
									
								
								context_test.go
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										13
									
								
								coverage.sh
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								coverage.sh
									
									
									
									
									
								
							| @ -1,13 +0,0 @@ | |||||||
| #!/usr/bin/env bash |  | ||||||
| 
 |  | ||||||
| set -e |  | ||||||
| 
 |  | ||||||
| echo "mode: count" > coverage.out |  | ||||||
| 
 |  | ||||||
| for d in $(go list ./... | grep -E 'gin$|binding$|render$'); do |  | ||||||
|     go test -v -covermode=count -coverprofile=profile.out $d |  | ||||||
|     if [ -f profile.out ]; then |  | ||||||
|         cat profile.out | grep -v "mode:" >> coverage.out |  | ||||||
|         rm profile.out |  | ||||||
|     fi |  | ||||||
| done |  | ||||||
							
								
								
									
										38
									
								
								debug.go
									
									
									
									
									
								
							
							
						
						
									
										38
									
								
								debug.go
									
									
									
									
									
								
							| @ -6,13 +6,15 @@ package gin | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
| 	"html/template" | 	"html/template" | ||||||
| 	"log" | 	"os" | ||||||
|  | 	"runtime" | ||||||
|  | 	"strconv" | ||||||
|  | 	"strings" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func init() { | const ginSupportMinGoVer = 6 | ||||||
| 	log.SetFlags(0) |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| // IsDebugging returns true if the framework is running in debug mode. | // IsDebugging returns true if the framework is running in debug mode. | ||||||
| // Use SetMode(gin.ReleaseMode) to disable debug mode. | // Use SetMode(gin.ReleaseMode) to disable debug mode. | ||||||
| @ -20,11 +22,18 @@ func IsDebugging() bool { | |||||||
| 	return ginMode == debugCode | 	return ginMode == debugCode | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // DebugPrintRouteFunc indicates debug log output format. | ||||||
|  | var DebugPrintRouteFunc func(httpMethod, absolutePath, handlerName string, nuHandlers int) | ||||||
|  | 
 | ||||||
| func debugPrintRoute(httpMethod, absolutePath string, handlers HandlersChain) { | func debugPrintRoute(httpMethod, absolutePath string, handlers HandlersChain) { | ||||||
| 	if IsDebugging() { | 	if IsDebugging() { | ||||||
| 		nuHandlers := len(handlers) | 		nuHandlers := len(handlers) | ||||||
| 		handlerName := nameOfFunction(handlers.Last()) | 		handlerName := nameOfFunction(handlers.Last()) | ||||||
| 		debugPrint("%-6s %-25s --> %s (%d handlers)\n", httpMethod, absolutePath, handlerName, nuHandlers) | 		if DebugPrintRouteFunc == nil { | ||||||
|  | 			debugPrint("%-6s %-25s --> %s (%d handlers)\n", httpMethod, absolutePath, handlerName, nuHandlers) | ||||||
|  | 		} else { | ||||||
|  | 			DebugPrintRouteFunc(httpMethod, absolutePath, handlerName, nuHandlers) | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -42,11 +51,28 @@ func debugPrintLoadTemplate(tmpl *template.Template) { | |||||||
| 
 | 
 | ||||||
| func debugPrint(format string, values ...interface{}) { | func debugPrint(format string, values ...interface{}) { | ||||||
| 	if IsDebugging() { | 	if IsDebugging() { | ||||||
| 		log.Printf("[GIN-debug] "+format, values...) | 		if !strings.HasSuffix(format, "\n") { | ||||||
|  | 			format += "\n" | ||||||
|  | 		} | ||||||
|  | 		fmt.Fprintf(os.Stderr, "[GIN-debug] "+format, values...) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func getMinVer(v string) (uint64, error) { | ||||||
|  | 	first := strings.IndexByte(v, '.') | ||||||
|  | 	last := strings.LastIndexByte(v, '.') | ||||||
|  | 	if first == last { | ||||||
|  | 		return strconv.ParseUint(v[first+1:], 10, 64) | ||||||
|  | 	} | ||||||
|  | 	return strconv.ParseUint(v[first+1:last], 10, 64) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func debugPrintWARNINGDefault() { | func debugPrintWARNINGDefault() { | ||||||
|  | 	if v, e := getMinVer(runtime.Version()); e == nil && v <= ginSupportMinGoVer { | ||||||
|  | 		debugPrint(`[WARNING] Now Gin requires Go 1.6 or later and Go 1.7 will be required soon. | ||||||
|  | 
 | ||||||
|  | `) | ||||||
|  | 	} | ||||||
| 	debugPrint(`[WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached. | 	debugPrint(`[WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached. | ||||||
| 
 | 
 | ||||||
| `) | `) | ||||||
|  | |||||||
							
								
								
									
										158
									
								
								debug_test.go
									
									
									
									
									
								
							
							
						
						
									
										158
									
								
								debug_test.go
									
									
									
									
									
								
							| @ -11,6 +11,8 @@ import ( | |||||||
| 	"io" | 	"io" | ||||||
| 	"log" | 	"log" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"runtime" | ||||||
|  | 	"sync" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| @ -30,86 +32,122 @@ func TestIsDebugging(t *testing.T) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrint(t *testing.T) { | func TestDebugPrint(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		SetMode(ReleaseMode) | ||||||
| 
 | 		debugPrint("DEBUG this!") | ||||||
| 	SetMode(ReleaseMode) | 		SetMode(TestMode) | ||||||
| 	debugPrint("DEBUG this!") | 		debugPrint("DEBUG this!") | ||||||
| 	SetMode(TestMode) | 		SetMode(DebugMode) | ||||||
| 	debugPrint("DEBUG this!") | 		debugPrint("these are %d %s", 2, "error messages") | ||||||
| 	assert.Empty(t, w.String()) | 		SetMode(TestMode) | ||||||
| 
 | 	}) | ||||||
| 	SetMode(DebugMode) | 	assert.Equal(t, "[GIN-debug] these are 2 error messages\n", re) | ||||||
| 	debugPrint("these are %d %s\n", 2, "error messages") |  | ||||||
| 	assert.Equal(t, "[GIN-debug] these are 2 error messages\n", w.String()) |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintError(t *testing.T) { | func TestDebugPrintError(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		debugPrintError(nil) | ||||||
| 
 | 		debugPrintError(errors.New("this is an error")) | ||||||
| 	SetMode(DebugMode) | 		SetMode(TestMode) | ||||||
| 	debugPrintError(nil) | 	}) | ||||||
| 	assert.Empty(t, w.String()) | 	assert.Equal(t, "[GIN-debug] [ERROR] this is an error\n", re) | ||||||
| 
 |  | ||||||
| 	debugPrintError(errors.New("this is an error")) |  | ||||||
| 	assert.Equal(t, "[GIN-debug] [ERROR] this is an error\n", w.String()) |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintRoutes(t *testing.T) { | func TestDebugPrintRoutes(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		debugPrintRoute("GET", "/path/to/route/:param", HandlersChain{func(c *Context) {}, handlerNameTest}) | ||||||
| 
 | 		SetMode(TestMode) | ||||||
| 	debugPrintRoute("GET", "/path/to/route/:param", HandlersChain{func(c *Context) {}, handlerNameTest}) | 	}) | ||||||
| 	assert.Regexp(t, `^\[GIN-debug\] GET    /path/to/route/:param     --> (.*/vendor/)?github.com/gin-gonic/gin.handlerNameTest \(2 handlers\)\n$`, w.String()) | 	assert.Regexp(t, `^\[GIN-debug\] GET    /path/to/route/:param     --> (.*/vendor/)?github.com/gin-gonic/gin.handlerNameTest \(2 handlers\)\n$`, re) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintLoadTemplate(t *testing.T) { | func TestDebugPrintLoadTemplate(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		templ := template.Must(template.New("").Delims("{[{", "}]}").ParseGlob("./testdata/template/hello.tmpl")) | ||||||
| 
 | 		debugPrintLoadTemplate(templ) | ||||||
| 	templ := template.Must(template.New("").Delims("{[{", "}]}").ParseGlob("./fixtures/basic/hello.tmpl")) | 		SetMode(TestMode) | ||||||
| 	debugPrintLoadTemplate(templ) | 	}) | ||||||
| 	assert.Regexp(t, `^\[GIN-debug\] Loaded HTML Templates \(2\): \n(\t- \n|\t- hello\.tmpl\n){2}\n`, w.String()) | 	assert.Regexp(t, `^\[GIN-debug\] Loaded HTML Templates \(2\): \n(\t- \n|\t- hello\.tmpl\n){2}\n`, re) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintWARNINGSetHTMLTemplate(t *testing.T) { | func TestDebugPrintWARNINGSetHTMLTemplate(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		debugPrintWARNINGSetHTMLTemplate() | ||||||
| 
 | 		SetMode(TestMode) | ||||||
| 	debugPrintWARNINGSetHTMLTemplate() | 	}) | ||||||
| 	assert.Equal(t, "[GIN-debug] [WARNING] Since SetHTMLTemplate() is NOT thread-safe. It should only be called\nat initialization. ie. before any route is registered or the router is listening in a socket:\n\n\trouter := gin.Default()\n\trouter.SetHTMLTemplate(template) // << good place\n\n", w.String()) | 	assert.Equal(t, "[GIN-debug] [WARNING] Since SetHTMLTemplate() is NOT thread-safe. It should only be called\nat initialization. ie. before any route is registered or the router is listening in a socket:\n\n\trouter := gin.Default()\n\trouter.SetHTMLTemplate(template) // << good place\n\n", re) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintWARNINGDefault(t *testing.T) { | func TestDebugPrintWARNINGDefault(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		debugPrintWARNINGDefault() | ||||||
| 
 | 		SetMode(TestMode) | ||||||
| 	debugPrintWARNINGDefault() | 	}) | ||||||
| 	assert.Equal(t, "[GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.\n\n", w.String()) | 	m, e := getMinVer(runtime.Version()) | ||||||
|  | 	if e == nil && m <= ginSupportMinGoVer { | ||||||
|  | 		assert.Equal(t, "[GIN-debug] [WARNING] Now Gin requires Go 1.6 or later and Go 1.7 will be required soon.\n\n[GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.\n\n", re) | ||||||
|  | 	} else { | ||||||
|  | 		assert.Equal(t, "[GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.\n\n", re) | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestDebugPrintWARNINGNew(t *testing.T) { | func TestDebugPrintWARNINGNew(t *testing.T) { | ||||||
| 	var w bytes.Buffer | 	re := captureOutput(t, func() { | ||||||
| 	setup(&w) | 		SetMode(DebugMode) | ||||||
| 	defer teardown() | 		debugPrintWARNINGNew() | ||||||
| 
 | 		SetMode(TestMode) | ||||||
| 	debugPrintWARNINGNew() | 	}) | ||||||
| 	assert.Equal(t, "[GIN-debug] [WARNING] Running in \"debug\" mode. Switch to \"release\" mode in production.\n - using env:\texport GIN_MODE=release\n - using code:\tgin.SetMode(gin.ReleaseMode)\n\n", w.String()) | 	assert.Equal(t, "[GIN-debug] [WARNING] Running in \"debug\" mode. Switch to \"release\" mode in production.\n - using env:\texport GIN_MODE=release\n - using code:\tgin.SetMode(gin.ReleaseMode)\n\n", re) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func setup(w io.Writer) { | func captureOutput(t *testing.T, f func()) string { | ||||||
| 	SetMode(DebugMode) | 	reader, writer, err := os.Pipe() | ||||||
| 	log.SetOutput(w) | 	if err != nil { | ||||||
|  | 		panic(err) | ||||||
|  | 	} | ||||||
|  | 	stdout := os.Stdout | ||||||
|  | 	stderr := os.Stderr | ||||||
|  | 	defer func() { | ||||||
|  | 		os.Stdout = stdout | ||||||
|  | 		os.Stderr = stderr | ||||||
|  | 		log.SetOutput(os.Stderr) | ||||||
|  | 	}() | ||||||
|  | 	os.Stdout = writer | ||||||
|  | 	os.Stderr = writer | ||||||
|  | 	log.SetOutput(writer) | ||||||
|  | 	out := make(chan string) | ||||||
|  | 	wg := new(sync.WaitGroup) | ||||||
|  | 	wg.Add(1) | ||||||
|  | 	go func() { | ||||||
|  | 		var buf bytes.Buffer | ||||||
|  | 		wg.Done() | ||||||
|  | 		_, err := io.Copy(&buf, reader) | ||||||
|  | 		assert.NoError(t, err) | ||||||
|  | 		out <- buf.String() | ||||||
|  | 	}() | ||||||
|  | 	wg.Wait() | ||||||
|  | 	f() | ||||||
|  | 	writer.Close() | ||||||
|  | 	return <-out | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func teardown() { | func TestGetMinVer(t *testing.T) { | ||||||
| 	SetMode(TestMode) | 	var m uint64 | ||||||
| 	log.SetOutput(os.Stdout) | 	var e error | ||||||
|  | 	_, e = getMinVer("go1") | ||||||
|  | 	assert.NotNil(t, e) | ||||||
|  | 	m, e = getMinVer("go1.1") | ||||||
|  | 	assert.Equal(t, uint64(1), m) | ||||||
|  | 	assert.Nil(t, e) | ||||||
|  | 	m, e = getMinVer("go1.1.1") | ||||||
|  | 	assert.Nil(t, e) | ||||||
|  | 	assert.Equal(t, uint64(1), m) | ||||||
|  | 	_, e = getMinVer("go1.1.1.1") | ||||||
|  | 	assert.NotNil(t, e) | ||||||
| } | } | ||||||
|  | |||||||
| @ -24,7 +24,9 @@ func TestBindWith(t *testing.T) { | |||||||
| 		Foo string `form:"foo"` | 		Foo string `form:"foo"` | ||||||
| 		Bar string `form:"bar"` | 		Bar string `form:"bar"` | ||||||
| 	} | 	} | ||||||
| 	assert.NoError(t, c.BindWith(&obj, binding.Form)) | 	captureOutput(t, func() { | ||||||
|  | 		assert.NoError(t, c.BindWith(&obj, binding.Form)) | ||||||
|  | 	}) | ||||||
| 	assert.Equal(t, "foo", obj.Bar) | 	assert.Equal(t, "foo", obj.Bar) | ||||||
| 	assert.Equal(t, "bar", obj.Foo) | 	assert.Equal(t, "bar", obj.Foo) | ||||||
| 	assert.Equal(t, 0, w.Body.Len()) | 	assert.Equal(t, 0, w.Body.Len()) | ||||||
|  | |||||||
							
								
								
									
										137
									
								
								docs/how-to-build-an-effective-middleware.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										137
									
								
								docs/how-to-build-an-effective-middleware.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,137 @@ | |||||||
|  | # How to build one effective middleware? | ||||||
|  | 
 | ||||||
|  | ## Consitituent part | ||||||
|  | 
 | ||||||
|  | The middleware has two parts: | ||||||
|  | 
 | ||||||
|  |   - part one is what is executed once, when you initialize your middleware. That's where you set up all the global objects, logicals etc. Everything that happens one per application lifetime. | ||||||
|  | 
 | ||||||
|  |   - part two is what executes on every request. For example, a database middleware you simply inject your "global" database object into the context. Once it's inside the context, you can retrieve it from within other middlewares and your handler function. | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | func funcName(params string) gin.HandlerFunc { | ||||||
|  |     // <--- | ||||||
|  |     // This is part one | ||||||
|  |     // ---> | ||||||
|  |     // The follow code is an example | ||||||
|  |     if err := check(params); err != nil { | ||||||
|  |         panic(err) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     return func(c *gin.Context) { | ||||||
|  |         // <--- | ||||||
|  |         // This is part two | ||||||
|  |         // ---> | ||||||
|  |         // The follow code is an example | ||||||
|  |         c.Set("TestVar", params) | ||||||
|  |         c.Next()     | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ## Execution process | ||||||
|  | 
 | ||||||
|  | Firstly, we have the follow example code: | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | func main() { | ||||||
|  | 	router := gin.Default() | ||||||
|  | 
 | ||||||
|  | 	router.Use(globalMiddleware()) | ||||||
|  | 
 | ||||||
|  | 	router.GET("/rest/n/api/*some", mid1(), mid2(), handler) | ||||||
|  | 
 | ||||||
|  | 	router.Run() | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func globalMiddleware() gin.HandlerFunc { | ||||||
|  | 	fmt.Println("globalMiddleware...1") | ||||||
|  | 
 | ||||||
|  | 	return func(c *gin.Context) { | ||||||
|  | 		fmt.Println("globalMiddleware...2") | ||||||
|  | 		c.Next() | ||||||
|  | 		fmt.Println("globalMiddleware...3") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func handler(c *gin.Context) { | ||||||
|  | 	fmt.Println("exec handler.") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func mid1() gin.HandlerFunc { | ||||||
|  | 	fmt.Println("mid1...1") | ||||||
|  | 
 | ||||||
|  | 	return func(c *gin.Context) { | ||||||
|  | 
 | ||||||
|  | 		fmt.Println("mid1...2") | ||||||
|  | 		c.Next() | ||||||
|  | 		fmt.Println("mid1...3") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func mid2() gin.HandlerFunc { | ||||||
|  | 	fmt.Println("mid2...1") | ||||||
|  | 
 | ||||||
|  | 	return func(c *gin.Context) { | ||||||
|  | 		fmt.Println("mid2...2") | ||||||
|  | 		c.Next() | ||||||
|  | 		fmt.Println("mid2...3") | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | According to [Consitituent part](#consitituent-part) said, when we run the gin process, **part one** will execute firstly and will print the follow information: | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | globalMiddleware...1 | ||||||
|  | mid1...1 | ||||||
|  | mid2...1 | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | And init order are: | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | globalMiddleware...1 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid1...1 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid2...1 | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | When we curl one request `curl -v localhost:8080/rest/n/api/some`, **part two** will execute their middleware and output the following information: | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | globalMiddleware...2 | ||||||
|  | mid1...2 | ||||||
|  | mid2...2 | ||||||
|  | exec handler. | ||||||
|  | mid2...3 | ||||||
|  | mid1...3 | ||||||
|  | globalMiddleware...3 | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | In other words, run order are: | ||||||
|  | 
 | ||||||
|  | ```go | ||||||
|  | globalMiddleware...2 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid1...2 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid2...2 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | exec handler. | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid2...3 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | mid1...3 | ||||||
|  |     | | ||||||
|  |     v | ||||||
|  | globalMiddleware...3 | ||||||
|  | ``` | ||||||
							
								
								
									
										26
									
								
								errors.go
									
									
									
									
									
								
							
							
						
						
									
										26
									
								
								errors.go
									
									
									
									
									
								
							| @ -9,21 +9,28 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"reflect" | 	"reflect" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin/json" | 	"github.com/gin-gonic/gin/internal/json" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // ErrorType is an unsigned 64-bit error code as defined in the gin spec. | ||||||
| type ErrorType uint64 | type ErrorType uint64 | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	ErrorTypeBind    ErrorType = 1 << 63 // used when c.Bind() fails | 	// ErrorTypeBind is used when Context.Bind() fails. | ||||||
| 	ErrorTypeRender  ErrorType = 1 << 62 // used when c.Render() fails | 	ErrorTypeBind ErrorType = 1 << 63 | ||||||
|  | 	// ErrorTypeRender is used when Context.Render() fails. | ||||||
|  | 	ErrorTypeRender ErrorType = 1 << 62 | ||||||
|  | 	// ErrorTypePrivate indicates a private error. | ||||||
| 	ErrorTypePrivate ErrorType = 1 << 0 | 	ErrorTypePrivate ErrorType = 1 << 0 | ||||||
| 	ErrorTypePublic  ErrorType = 1 << 1 | 	// ErrorTypePublic indicates a public error. | ||||||
| 
 | 	ErrorTypePublic ErrorType = 1 << 1 | ||||||
|  | 	// ErrorTypeAny indicates any other error. | ||||||
| 	ErrorTypeAny ErrorType = 1<<64 - 1 | 	ErrorTypeAny ErrorType = 1<<64 - 1 | ||||||
| 	ErrorTypeNu            = 2 | 	// ErrorTypeNu indicates any other error. | ||||||
|  | 	ErrorTypeNu = 2 | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Error represents a error's specification. | ||||||
| type Error struct { | type Error struct { | ||||||
| 	Err  error | 	Err  error | ||||||
| 	Type ErrorType | 	Type ErrorType | ||||||
| @ -34,16 +41,19 @@ type errorMsgs []*Error | |||||||
| 
 | 
 | ||||||
| var _ error = &Error{} | var _ error = &Error{} | ||||||
| 
 | 
 | ||||||
|  | // SetType sets the error's type. | ||||||
| func (msg *Error) SetType(flags ErrorType) *Error { | func (msg *Error) SetType(flags ErrorType) *Error { | ||||||
| 	msg.Type = flags | 	msg.Type = flags | ||||||
| 	return msg | 	return msg | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // SetMeta sets the error's meta data. | ||||||
| func (msg *Error) SetMeta(data interface{}) *Error { | func (msg *Error) SetMeta(data interface{}) *Error { | ||||||
| 	msg.Meta = data | 	msg.Meta = data | ||||||
| 	return msg | 	return msg | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // JSON creates a properly formated JSON | ||||||
| func (msg *Error) JSON() interface{} { | func (msg *Error) JSON() interface{} { | ||||||
| 	json := H{} | 	json := H{} | ||||||
| 	if msg.Meta != nil { | 	if msg.Meta != nil { | ||||||
| @ -70,11 +80,12 @@ func (msg *Error) MarshalJSON() ([]byte, error) { | |||||||
| 	return json.Marshal(msg.JSON()) | 	return json.Marshal(msg.JSON()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Error implements the error interface | // Error implements the error interface. | ||||||
| func (msg Error) Error() string { | func (msg Error) Error() string { | ||||||
| 	return msg.Err.Error() | 	return msg.Err.Error() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // IsType judges one error. | ||||||
| func (msg *Error) IsType(flags ErrorType) bool { | func (msg *Error) IsType(flags ErrorType) bool { | ||||||
| 	return (msg.Type & flags) > 0 | 	return (msg.Type & flags) > 0 | ||||||
| } | } | ||||||
| @ -138,6 +149,7 @@ func (a errorMsgs) JSON() interface{} { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // MarshalJSON implements the json.Marshaller interface. | ||||||
| func (a errorMsgs) MarshalJSON() ([]byte, error) { | func (a errorMsgs) MarshalJSON() ([]byte, error) { | ||||||
| 	return json.Marshal(a.JSON()) | 	return json.Marshal(a.JSON()) | ||||||
| } | } | ||||||
|  | |||||||
| @ -8,7 +8,7 @@ import ( | |||||||
| 	"errors" | 	"errors" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin/json" | 	"github.com/gin-gonic/gin/internal/json" | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| @ -19,47 +19,47 @@ func TestError(t *testing.T) { | |||||||
| 		Type: ErrorTypePrivate, | 		Type: ErrorTypePrivate, | ||||||
| 	} | 	} | ||||||
| 	assert.Equal(t, err.Error(), baseError.Error()) | 	assert.Equal(t, err.Error(), baseError.Error()) | ||||||
| 	assert.Equal(t, err.JSON(), H{"error": baseError.Error()}) | 	assert.Equal(t, H{"error": baseError.Error()}, err.JSON()) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, err.SetType(ErrorTypePublic), err) | 	assert.Equal(t, err.SetType(ErrorTypePublic), err) | ||||||
| 	assert.Equal(t, err.Type, ErrorTypePublic) | 	assert.Equal(t, ErrorTypePublic, err.Type) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, err.SetMeta("some data"), err) | 	assert.Equal(t, err.SetMeta("some data"), err) | ||||||
| 	assert.Equal(t, err.Meta, "some data") | 	assert.Equal(t, "some data", err.Meta) | ||||||
| 	assert.Equal(t, err.JSON(), H{ | 	assert.Equal(t, H{ | ||||||
| 		"error": baseError.Error(), | 		"error": baseError.Error(), | ||||||
| 		"meta":  "some data", | 		"meta":  "some data", | ||||||
| 	}) | 	}, err.JSON()) | ||||||
| 
 | 
 | ||||||
| 	jsonBytes, _ := json.Marshal(err) | 	jsonBytes, _ := json.Marshal(err) | ||||||
| 	assert.Equal(t, "{\"error\":\"test error\",\"meta\":\"some data\"}", string(jsonBytes)) | 	assert.Equal(t, "{\"error\":\"test error\",\"meta\":\"some data\"}", string(jsonBytes)) | ||||||
| 
 | 
 | ||||||
| 	err.SetMeta(H{ | 	err.SetMeta(H{ // nolint: errcheck | ||||||
| 		"status": "200", | 		"status": "200", | ||||||
| 		"data":   "some data", | 		"data":   "some data", | ||||||
| 	}) | 	}) | ||||||
| 	assert.Equal(t, err.JSON(), H{ | 	assert.Equal(t, H{ | ||||||
| 		"error":  baseError.Error(), | 		"error":  baseError.Error(), | ||||||
| 		"status": "200", | 		"status": "200", | ||||||
| 		"data":   "some data", | 		"data":   "some data", | ||||||
| 	}) | 	}, err.JSON()) | ||||||
| 
 | 
 | ||||||
| 	err.SetMeta(H{ | 	err.SetMeta(H{ // nolint: errcheck | ||||||
| 		"error":  "custom error", | 		"error":  "custom error", | ||||||
| 		"status": "200", | 		"status": "200", | ||||||
| 		"data":   "some data", | 		"data":   "some data", | ||||||
| 	}) | 	}) | ||||||
| 	assert.Equal(t, err.JSON(), H{ | 	assert.Equal(t, H{ | ||||||
| 		"error":  "custom error", | 		"error":  "custom error", | ||||||
| 		"status": "200", | 		"status": "200", | ||||||
| 		"data":   "some data", | 		"data":   "some data", | ||||||
| 	}) | 	}, err.JSON()) | ||||||
| 
 | 
 | ||||||
| 	type customError struct { | 	type customError struct { | ||||||
| 		status string | 		status string | ||||||
| 		data   string | 		data   string | ||||||
| 	} | 	} | ||||||
| 	err.SetMeta(customError{status: "200", data: "other data"}) | 	err.SetMeta(customError{status: "200", data: "other data"}) // nolint: errcheck | ||||||
| 	assert.Equal(t, customError{status: "200", data: "other data"}, err.JSON()) | 	assert.Equal(t, customError{status: "200", data: "other data"}, err.JSON()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,7 +1,8 @@ | |||||||
| # Guide to run Gin under App Engine LOCAL Development Server | # Guide to run Gin under App Engine LOCAL Development Server | ||||||
| 
 | 
 | ||||||
| 1. Download, install and setup Go in your computer. (That includes setting your `$GOPATH`.) | 1. Download, install and setup Go in your computer. (That includes setting your `$GOPATH`.) | ||||||
| 2. Download SDK for your platform from here: `https://developers.google.com/appengine/downloads?hl=es#Google_App_Engine_SDK_for_Go` | 2. Download SDK for your platform from [here](https://cloud.google.com/appengine/docs/standard/go/download): `https://cloud.google.com/appengine/docs/standard/go/download` | ||||||
| 3. Download Gin source code using: `$ go get github.com/gin-gonic/gin` | 3. Download Gin source code using: `$ go get github.com/gin-gonic/gin` | ||||||
| 4. Navigate to examples folder: `$ cd $GOPATH/src/github.com/gin-gonic/gin/examples/` | 4. Navigate to examples folder: `$ cd $GOPATH/src/github.com/gin-gonic/gin/examples/app-engine/` | ||||||
| 5. Run it: `$ goapp serve app-engine/` | 5. Run it: `$ dev_appserver.py .` (notice that you have to run this script by Python2) | ||||||
|  | 
 | ||||||
|  | |||||||
							
								
								
									
										33
									
								
								examples/assets-in-binary/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								examples/assets-in-binary/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,33 @@ | |||||||
|  | # Building a single binary containing templates | ||||||
|  | 
 | ||||||
|  | This is a complete example to create a single binary with the | ||||||
|  | [gin-gonic/gin][gin] Web Server with HTML templates. | ||||||
|  | 
 | ||||||
|  | [gin]: https://github.com/gin-gonic/gin | ||||||
|  | 
 | ||||||
|  | ## How to use | ||||||
|  | 
 | ||||||
|  | ### Prepare Packages | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | go get github.com/gin-gonic/gin | ||||||
|  | go get github.com/jessevdk/go-assets-builder | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Generate assets.go | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | go-assets-builder html -o assets.go | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Build the server | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | go build -o assets-in-binary | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Run | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | ./assets-in-binary | ||||||
|  | ``` | ||||||
							
								
								
									
										34
									
								
								examples/assets-in-binary/assets.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								examples/assets-in-binary/assets.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"time" | ||||||
|  | 
 | ||||||
|  | 	"github.com/jessevdk/go-assets" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | var _Assetsbfa8d115ce0617d89507412d5393a462f8e9b003 = "<!doctype html>\n<body>\n  <p>Can you see this? → {{.Bar}}</p>\n</body>\n" | ||||||
|  | var _Assets3737a75b5254ed1f6d588b40a3449721f9ea86c2 = "<!doctype html>\n<body>\n  <p>Hello, {{.Foo}}</p>\n</body>\n" | ||||||
|  | 
 | ||||||
|  | // Assets returns go-assets FileSystem | ||||||
|  | var Assets = assets.NewFileSystem(map[string][]string{"/": {"html"}, "/html": {"bar.tmpl", "index.tmpl"}}, map[string]*assets.File{ | ||||||
|  | 	"/": { | ||||||
|  | 		Path:     "/", | ||||||
|  | 		FileMode: 0x800001ed, | ||||||
|  | 		Mtime:    time.Unix(1524365738, 1524365738517125470), | ||||||
|  | 		Data:     nil, | ||||||
|  | 	}, "/html": { | ||||||
|  | 		Path:     "/html", | ||||||
|  | 		FileMode: 0x800001ed, | ||||||
|  | 		Mtime:    time.Unix(1524365491, 1524365491289799093), | ||||||
|  | 		Data:     nil, | ||||||
|  | 	}, "/html/bar.tmpl": { | ||||||
|  | 		Path:     "/html/bar.tmpl", | ||||||
|  | 		FileMode: 0x1a4, | ||||||
|  | 		Mtime:    time.Unix(1524365491, 1524365491289611557), | ||||||
|  | 		Data:     []byte(_Assetsbfa8d115ce0617d89507412d5393a462f8e9b003), | ||||||
|  | 	}, "/html/index.tmpl": { | ||||||
|  | 		Path:     "/html/index.tmpl", | ||||||
|  | 		FileMode: 0x1a4, | ||||||
|  | 		Mtime:    time.Unix(1524365491, 1524365491289995821), | ||||||
|  | 		Data:     []byte(_Assets3737a75b5254ed1f6d588b40a3449721f9ea86c2), | ||||||
|  | 	}}, "") | ||||||
							
								
								
									
										4
									
								
								examples/assets-in-binary/html/bar.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								examples/assets-in-binary/html/bar.tmpl
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | <!doctype html> | ||||||
|  | <body> | ||||||
|  |   <p>Can you see this? → {{.Bar}}</p> | ||||||
|  | </body> | ||||||
							
								
								
									
										4
									
								
								examples/assets-in-binary/html/index.tmpl
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								examples/assets-in-binary/html/index.tmpl
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | <!doctype html> | ||||||
|  | <body> | ||||||
|  |   <p>Hello, {{.Foo}}</p> | ||||||
|  | </body> | ||||||
							
								
								
									
										48
									
								
								examples/assets-in-binary/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								examples/assets-in-binary/main.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,48 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"html/template" | ||||||
|  | 	"io/ioutil" | ||||||
|  | 	"net/http" | ||||||
|  | 	"strings" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	r := gin.New() | ||||||
|  | 	t, err := loadTemplate() | ||||||
|  | 	if err != nil { | ||||||
|  | 		panic(err) | ||||||
|  | 	} | ||||||
|  | 	r.SetHTMLTemplate(t) | ||||||
|  | 	r.GET("/", func(c *gin.Context) { | ||||||
|  | 		c.HTML(http.StatusOK, "/html/index.tmpl", gin.H{ | ||||||
|  | 			"Foo": "World", | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  | 	r.GET("/bar", func(c *gin.Context) { | ||||||
|  | 		c.HTML(http.StatusOK, "/html/bar.tmpl", gin.H{ | ||||||
|  | 			"Bar": "World", | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  | 	r.Run(":8080") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func loadTemplate() (*template.Template, error) { | ||||||
|  | 	t := template.New("") | ||||||
|  | 	for name, file := range Assets.Files { | ||||||
|  | 		if file.IsDir() || !strings.HasSuffix(name, ".tmpl") { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		h, err := ioutil.ReadAll(file) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		t, err = t.New(name).Parse(string(h)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return t, nil | ||||||
|  | } | ||||||
| @ -1,10 +1,12 @@ | |||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var DB = make(map[string]string) | var db = make(map[string]string) | ||||||
| 
 | 
 | ||||||
| func setupRouter() *gin.Engine { | func setupRouter() *gin.Engine { | ||||||
| 	// Disable Console Color | 	// Disable Console Color | ||||||
| @ -13,17 +15,17 @@ func setupRouter() *gin.Engine { | |||||||
| 
 | 
 | ||||||
| 	// Ping test | 	// Ping test | ||||||
| 	r.GET("/ping", func(c *gin.Context) { | 	r.GET("/ping", func(c *gin.Context) { | ||||||
| 		c.String(200, "pong") | 		c.String(http.StatusOK, "pong") | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	// Get user value | 	// Get user value | ||||||
| 	r.GET("/user/:name", func(c *gin.Context) { | 	r.GET("/user/:name", func(c *gin.Context) { | ||||||
| 		user := c.Params.ByName("name") | 		user := c.Params.ByName("name") | ||||||
| 		value, ok := DB[user] | 		value, ok := db[user] | ||||||
| 		if ok { | 		if ok { | ||||||
| 			c.JSON(200, gin.H{"user": user, "value": value}) | 			c.JSON(http.StatusOK, gin.H{"user": user, "value": value}) | ||||||
| 		} else { | 		} else { | ||||||
| 			c.JSON(200, gin.H{"user": user, "status": "no value"}) | 			c.JSON(http.StatusOK, gin.H{"user": user, "status": "no value"}) | ||||||
| 		} | 		} | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| @ -48,8 +50,8 @@ func setupRouter() *gin.Engine { | |||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if c.Bind(&json) == nil { | 		if c.Bind(&json) == nil { | ||||||
| 			DB[user] = json.Value | 			db[user] = json.Value | ||||||
| 			c.JSON(200, gin.H{"status": "ok"}) | 			c.JSON(http.StatusOK, gin.H{"status": "ok"}) | ||||||
| 		} | 		} | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -15,6 +15,6 @@ func TestPingRoute(t *testing.T) { | |||||||
| 	req, _ := http.NewRequest("GET", "/ping", nil) | 	req, _ := http.NewRequest("GET", "/ping", nil) | ||||||
| 	router.ServeHTTP(w, req) | 	router.ServeHTTP(w, req) | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, 200, w.Code) | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
| 	assert.Equal(t, "pong", w.Body.String()) | 	assert.Equal(t, "pong", w.Body.String()) | ||||||
| } | } | ||||||
|  | |||||||
| @ -10,6 +10,7 @@ import ( | |||||||
| 	"gopkg.in/go-playground/validator.v8" | 	"gopkg.in/go-playground/validator.v8" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Booking contains binded and validated data. | ||||||
| type Booking struct { | type Booking struct { | ||||||
| 	CheckIn  time.Time `form:"check_in" binding:"required,bookabledate" time_format:"2006-01-02"` | 	CheckIn  time.Time `form:"check_in" binding:"required,bookabledate" time_format:"2006-01-02"` | ||||||
| 	CheckOut time.Time `form:"check_out" binding:"required,gtfield=CheckIn" time_format:"2006-01-02"` | 	CheckOut time.Time `form:"check_out" binding:"required,gtfield=CheckIn" time_format:"2006-01-02"` | ||||||
| @ -30,7 +31,11 @@ func bookableDate( | |||||||
| 
 | 
 | ||||||
| func main() { | func main() { | ||||||
| 	route := gin.Default() | 	route := gin.Default() | ||||||
| 	binding.Validator.RegisterValidation("bookabledate", bookableDate) | 
 | ||||||
|  | 	if v, ok := binding.Validator.Engine().(*validator.Validate); ok { | ||||||
|  | 		v.RegisterValidation("bookabledate", bookableDate) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
| 	route.GET("/bookable", getBookable) | 	route.GET("/bookable", getBookable) | ||||||
| 	route.Run(":8085") | 	route.Run(":8085") | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,6 +1,8 @@ | |||||||
| package main | package main | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| 	"github.com/thinkerou/favicon" | 	"github.com/thinkerou/favicon" | ||||||
| ) | ) | ||||||
| @ -9,7 +11,7 @@ func main() { | |||||||
| 	app := gin.Default() | 	app := gin.Default() | ||||||
| 	app.Use(favicon.New("./favicon.ico")) | 	app.Use(favicon.New("./favicon.ico")) | ||||||
| 	app.GET("/ping", func(c *gin.Context) { | 	app.GET("/ping", func(c *gin.Context) { | ||||||
| 		c.String(200, "Hello favicon.") | 		c.String(http.StatusOK, "Hello favicon.") | ||||||
| 	}) | 	}) | ||||||
| 	app.Run(":8080") | 	app.Run(":8080") | ||||||
| } | } | ||||||
|  | |||||||
| @ -8,6 +8,7 @@ import ( | |||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 	"os/signal" | 	"os/signal" | ||||||
|  | 	"syscall" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| @ -27,15 +28,18 @@ func main() { | |||||||
| 
 | 
 | ||||||
| 	go func() { | 	go func() { | ||||||
| 		// service connections | 		// service connections | ||||||
| 		if err := srv.ListenAndServe(); err != nil { | 		if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { | ||||||
| 			log.Printf("listen: %s\n", err) | 			log.Fatalf("listen: %s\n", err) | ||||||
| 		} | 		} | ||||||
| 	}() | 	}() | ||||||
| 
 | 
 | ||||||
| 	// Wait for interrupt signal to gracefully shutdown the server with | 	// Wait for interrupt signal to gracefully shutdown the server with | ||||||
| 	// a timeout of 5 seconds. | 	// a timeout of 5 seconds. | ||||||
| 	quit := make(chan os.Signal) | 	quit := make(chan os.Signal) | ||||||
| 	signal.Notify(quit, os.Interrupt) | 	// kill (no param) default send syscanll.SIGTERM | ||||||
|  | 	// kill -2 is syscall.SIGINT | ||||||
|  | 	// kill -9 is syscall. SIGKILL but can"t be catch, so don't need add it | ||||||
|  | 	signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM) | ||||||
| 	<-quit | 	<-quit | ||||||
| 	log.Println("Shutdown Server ...") | 	log.Println("Shutdown Server ...") | ||||||
| 
 | 
 | ||||||
| @ -44,5 +48,10 @@ func main() { | |||||||
| 	if err := srv.Shutdown(ctx); err != nil { | 	if err := srv.Shutdown(ctx); err != nil { | ||||||
| 		log.Fatal("Server Shutdown:", err) | 		log.Fatal("Server Shutdown:", err) | ||||||
| 	} | 	} | ||||||
|  | 	// catching ctx.Done(). timeout of 5 seconds. | ||||||
|  | 	select { | ||||||
|  | 	case <-ctx.Done(): | ||||||
|  | 		log.Println("timeout of 5 seconds.") | ||||||
|  | 	} | ||||||
| 	log.Println("Server exiting") | 	log.Println("Server exiting") | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										19
									
								
								examples/grpc/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								examples/grpc/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,19 @@ | |||||||
|  | ## How to run this example | ||||||
|  | 
 | ||||||
|  | 1. run grpc server | ||||||
|  | 
 | ||||||
|  | ```sh | ||||||
|  | $ go run grpc/server.go | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | 2. run gin server | ||||||
|  | 
 | ||||||
|  | ```sh | ||||||
|  | $ go run gin/main.go | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | 3. use curl command to test it | ||||||
|  | 
 | ||||||
|  | ```sh | ||||||
|  | $ curl -v 'http://localhost:8052/rest/n/thinkerou' | ||||||
|  | ``` | ||||||
							
								
								
									
										46
									
								
								examples/grpc/gin/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								examples/grpc/gin/main.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,46 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"log" | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	pb "github.com/gin-gonic/gin/examples/grpc/pb" | ||||||
|  | 	"google.golang.org/grpc" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	// Set up a connection to the server. | ||||||
|  | 	conn, err := grpc.Dial("localhost:50051", grpc.WithInsecure()) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatalf("did not connect: %v", err) | ||||||
|  | 	} | ||||||
|  | 	defer conn.Close() | ||||||
|  | 	client := pb.NewGreeterClient(conn) | ||||||
|  | 
 | ||||||
|  | 	// Set up a http server. | ||||||
|  | 	r := gin.Default() | ||||||
|  | 	r.GET("/rest/n/:name", func(c *gin.Context) { | ||||||
|  | 		name := c.Param("name") | ||||||
|  | 
 | ||||||
|  | 		// Contact the server and print out its response. | ||||||
|  | 		req := &pb.HelloRequest{Name: name} | ||||||
|  | 		res, err := client.SayHello(c, req) | ||||||
|  | 		if err != nil { | ||||||
|  | 			c.JSON(http.StatusInternalServerError, gin.H{ | ||||||
|  | 				"error": err.Error(), | ||||||
|  | 			}) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		c.JSON(http.StatusOK, gin.H{ | ||||||
|  | 			"result": fmt.Sprint(res.Message), | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	// Run http server | ||||||
|  | 	if err := r.Run(":8052"); err != nil { | ||||||
|  | 		log.Fatalf("could not run server: %v", err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										34
									
								
								examples/grpc/grpc/server.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										34
									
								
								examples/grpc/grpc/server.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,34 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"log" | ||||||
|  | 	"net" | ||||||
|  | 
 | ||||||
|  | 	pb "github.com/gin-gonic/gin/examples/grpc/pb" | ||||||
|  | 	"golang.org/x/net/context" | ||||||
|  | 	"google.golang.org/grpc" | ||||||
|  | 	"google.golang.org/grpc/reflection" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // server is used to implement helloworld.GreeterServer. | ||||||
|  | type server struct{} | ||||||
|  | 
 | ||||||
|  | // SayHello implements helloworld.GreeterServer | ||||||
|  | func (s *server) SayHello(ctx context.Context, in *pb.HelloRequest) (*pb.HelloReply, error) { | ||||||
|  | 	return &pb.HelloReply{Message: "Hello " + in.Name}, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	lis, err := net.Listen("tcp", ":50051") | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Fatalf("failed to listen: %v", err) | ||||||
|  | 	} | ||||||
|  | 	s := grpc.NewServer() | ||||||
|  | 	pb.RegisterGreeterServer(s, &server{}) | ||||||
|  | 
 | ||||||
|  | 	// Register reflection service on gRPC server. | ||||||
|  | 	reflection.Register(s) | ||||||
|  | 	if err := s.Serve(lis); err != nil { | ||||||
|  | 		log.Fatalf("failed to serve: %v", err) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										151
									
								
								examples/grpc/pb/helloworld.pb.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										151
									
								
								examples/grpc/pb/helloworld.pb.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,151 @@ | |||||||
|  | // Code generated by protoc-gen-go. | ||||||
|  | // source: helloworld.proto | ||||||
|  | // DO NOT EDIT! | ||||||
|  | 
 | ||||||
|  | /* | ||||||
|  | Package helloworld is a generated protocol buffer package. | ||||||
|  | 
 | ||||||
|  | It is generated from these files: | ||||||
|  | 	helloworld.proto | ||||||
|  | 
 | ||||||
|  | It has these top-level messages: | ||||||
|  | 	HelloRequest | ||||||
|  | 	HelloReply | ||||||
|  | */ | ||||||
|  | package helloworld | ||||||
|  | 
 | ||||||
|  | import proto "github.com/golang/protobuf/proto" | ||||||
|  | import fmt "fmt" | ||||||
|  | import math "math" | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	context "golang.org/x/net/context" | ||||||
|  | 	grpc "google.golang.org/grpc" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // Reference imports to suppress errors if they are not otherwise used. | ||||||
|  | var _ = proto.Marshal | ||||||
|  | var _ = fmt.Errorf | ||||||
|  | var _ = math.Inf | ||||||
|  | 
 | ||||||
|  | // This is a compile-time assertion to ensure that this generated file | ||||||
|  | // is compatible with the proto package it is being compiled against. | ||||||
|  | // A compilation error at this line likely means your copy of the | ||||||
|  | // proto package needs to be updated. | ||||||
|  | const _ = proto.ProtoPackageIsVersion2 // please upgrade the proto package | ||||||
|  | 
 | ||||||
|  | // The request message containing the user's name. | ||||||
|  | type HelloRequest struct { | ||||||
|  | 	Name string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (m *HelloRequest) Reset()                    { *m = HelloRequest{} } | ||||||
|  | func (m *HelloRequest) String() string            { return proto.CompactTextString(m) } | ||||||
|  | func (*HelloRequest) ProtoMessage()               {} | ||||||
|  | func (*HelloRequest) Descriptor() ([]byte, []int) { return fileDescriptor0, []int{0} } | ||||||
|  | 
 | ||||||
|  | // The response message containing the greetings | ||||||
|  | type HelloReply struct { | ||||||
|  | 	Message string `protobuf:"bytes,1,opt,name=message" json:"message,omitempty"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (m *HelloReply) Reset()                    { *m = HelloReply{} } | ||||||
|  | func (m *HelloReply) String() string            { return proto.CompactTextString(m) } | ||||||
|  | func (*HelloReply) ProtoMessage()               {} | ||||||
|  | func (*HelloReply) Descriptor() ([]byte, []int) { return fileDescriptor0, []int{1} } | ||||||
|  | 
 | ||||||
|  | func init() { | ||||||
|  | 	proto.RegisterType((*HelloRequest)(nil), "helloworld.HelloRequest") | ||||||
|  | 	proto.RegisterType((*HelloReply)(nil), "helloworld.HelloReply") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Reference imports to suppress errors if they are not otherwise used. | ||||||
|  | var _ context.Context | ||||||
|  | var _ grpc.ClientConn | ||||||
|  | 
 | ||||||
|  | // This is a compile-time assertion to ensure that this generated file | ||||||
|  | // is compatible with the grpc package it is being compiled against. | ||||||
|  | const _ = grpc.SupportPackageIsVersion4 | ||||||
|  | 
 | ||||||
|  | // Client API for Greeter service | ||||||
|  | 
 | ||||||
|  | type GreeterClient interface { | ||||||
|  | 	// Sends a greeting | ||||||
|  | 	SayHello(ctx context.Context, in *HelloRequest, opts ...grpc.CallOption) (*HelloReply, error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | type greeterClient struct { | ||||||
|  | 	cc *grpc.ClientConn | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func NewGreeterClient(cc *grpc.ClientConn) GreeterClient { | ||||||
|  | 	return &greeterClient{cc} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func (c *greeterClient) SayHello(ctx context.Context, in *HelloRequest, opts ...grpc.CallOption) (*HelloReply, error) { | ||||||
|  | 	out := new(HelloReply) | ||||||
|  | 	err := grpc.Invoke(ctx, "/helloworld.Greeter/SayHello", in, out, c.cc, opts...) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	return out, nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Server API for Greeter service | ||||||
|  | 
 | ||||||
|  | type GreeterServer interface { | ||||||
|  | 	// Sends a greeting | ||||||
|  | 	SayHello(context.Context, *HelloRequest) (*HelloReply, error) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func RegisterGreeterServer(s *grpc.Server, srv GreeterServer) { | ||||||
|  | 	s.RegisterService(&_Greeter_serviceDesc, srv) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func _Greeter_SayHello_Handler(srv interface{}, ctx context.Context, dec func(interface{}) error, interceptor grpc.UnaryServerInterceptor) (interface{}, error) { | ||||||
|  | 	in := new(HelloRequest) | ||||||
|  | 	if err := dec(in); err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  | 	if interceptor == nil { | ||||||
|  | 		return srv.(GreeterServer).SayHello(ctx, in) | ||||||
|  | 	} | ||||||
|  | 	info := &grpc.UnaryServerInfo{ | ||||||
|  | 		Server:     srv, | ||||||
|  | 		FullMethod: "/helloworld.Greeter/SayHello", | ||||||
|  | 	} | ||||||
|  | 	handler := func(ctx context.Context, req interface{}) (interface{}, error) { | ||||||
|  | 		return srv.(GreeterServer).SayHello(ctx, req.(*HelloRequest)) | ||||||
|  | 	} | ||||||
|  | 	return interceptor(ctx, in, info, handler) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var _Greeter_serviceDesc = grpc.ServiceDesc{ | ||||||
|  | 	ServiceName: "helloworld.Greeter", | ||||||
|  | 	HandlerType: (*GreeterServer)(nil), | ||||||
|  | 	Methods: []grpc.MethodDesc{ | ||||||
|  | 		{ | ||||||
|  | 			MethodName: "SayHello", | ||||||
|  | 			Handler:    _Greeter_SayHello_Handler, | ||||||
|  | 		}, | ||||||
|  | 	}, | ||||||
|  | 	Streams:  []grpc.StreamDesc{}, | ||||||
|  | 	Metadata: "helloworld.proto", | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func init() { proto.RegisterFile("helloworld.proto", fileDescriptor0) } | ||||||
|  | 
 | ||||||
|  | var fileDescriptor0 = []byte{ | ||||||
|  | 	// 174 bytes of a gzipped FileDescriptorProto | ||||||
|  | 	0x1f, 0x8b, 0x08, 0x00, 0x00, 0x09, 0x6e, 0x88, 0x02, 0xff, 0xe2, 0x12, 0xc8, 0x48, 0xcd, 0xc9, | ||||||
|  | 	0xc9, 0x2f, 0xcf, 0x2f, 0xca, 0x49, 0xd1, 0x2b, 0x28, 0xca, 0x2f, 0xc9, 0x17, 0xe2, 0x42, 0x88, | ||||||
|  | 	0x28, 0x29, 0x71, 0xf1, 0x78, 0x80, 0x78, 0x41, 0xa9, 0x85, 0xa5, 0xa9, 0xc5, 0x25, 0x42, 0x42, | ||||||
|  | 	0x5c, 0x2c, 0x79, 0x89, 0xb9, 0xa9, 0x12, 0x8c, 0x0a, 0x8c, 0x1a, 0x9c, 0x41, 0x60, 0xb6, 0x92, | ||||||
|  | 	0x1a, 0x17, 0x17, 0x54, 0x4d, 0x41, 0x4e, 0xa5, 0x90, 0x04, 0x17, 0x7b, 0x6e, 0x6a, 0x71, 0x71, | ||||||
|  | 	0x62, 0x3a, 0x4c, 0x11, 0x8c, 0x6b, 0xe4, 0xc9, 0xc5, 0xee, 0x5e, 0x94, 0x9a, 0x5a, 0x92, 0x5a, | ||||||
|  | 	0x24, 0x64, 0xc7, 0xc5, 0x11, 0x9c, 0x58, 0x09, 0xd6, 0x25, 0x24, 0xa1, 0x87, 0xe4, 0x02, 0x64, | ||||||
|  | 	0xcb, 0xa4, 0xc4, 0xb0, 0xc8, 0x00, 0xad, 0x50, 0x62, 0x70, 0x32, 0xe0, 0x92, 0xce, 0xcc, 0xd7, | ||||||
|  | 	0x4b, 0x2f, 0x2a, 0x48, 0xd6, 0x4b, 0xad, 0x48, 0xcc, 0x2d, 0xc8, 0x49, 0x2d, 0x46, 0x52, 0xeb, | ||||||
|  | 	0xc4, 0x0f, 0x56, 0x1c, 0x0e, 0x62, 0x07, 0x80, 0xbc, 0x14, 0xc0, 0x98, 0xc4, 0x06, 0xf6, 0x9b, | ||||||
|  | 	0x31, 0x20, 0x00, 0x00, 0xff, 0xff, 0x0f, 0xb7, 0xcd, 0xf2, 0xef, 0x00, 0x00, 0x00, | ||||||
|  | } | ||||||
							
								
								
									
										37
									
								
								examples/grpc/pb/helloworld.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								examples/grpc/pb/helloworld.proto
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | |||||||
|  | // Copyright 2015 gRPC authors. | ||||||
|  | // | ||||||
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  | // you may not use this file except in compliance with the License. | ||||||
|  | // You may obtain a copy of the License at | ||||||
|  | // | ||||||
|  | //     http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | // | ||||||
|  | // Unless required by applicable law or agreed to in writing, software | ||||||
|  | // distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  | // See the License for the specific language governing permissions and | ||||||
|  | // limitations under the License. | ||||||
|  | 
 | ||||||
|  | syntax = "proto3"; | ||||||
|  | 
 | ||||||
|  | option java_multiple_files = true; | ||||||
|  | option java_package = "io.grpc.examples.helloworld"; | ||||||
|  | option java_outer_classname = "HelloWorldProto"; | ||||||
|  | 
 | ||||||
|  | package helloworld; | ||||||
|  | 
 | ||||||
|  | // The greeting service definition. | ||||||
|  | service Greeter { | ||||||
|  |   // Sends a greeting | ||||||
|  |   rpc SayHello (HelloRequest) returns (HelloReply) {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // The request message containing the user's name. | ||||||
|  | message HelloRequest { | ||||||
|  |   string name = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // The response message containing the greetings | ||||||
|  | message HelloReply { | ||||||
|  |   string message = 1; | ||||||
|  | } | ||||||
							
								
								
									
										1
									
								
								examples/http-pusher/assets/app.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								examples/http-pusher/assets/app.js
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | console.log("http2 pusher"); | ||||||
							
								
								
									
										41
									
								
								examples/http-pusher/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								examples/http-pusher/main.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"html/template" | ||||||
|  | 	"log" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | var html = template.Must(template.New("https").Parse(` | ||||||
|  | <html> | ||||||
|  | <head> | ||||||
|  |   <title>Https Test</title> | ||||||
|  |   <script src="/assets/app.js"></script> | ||||||
|  | </head> | ||||||
|  | <body> | ||||||
|  |   <h1 style="color:red;">Welcome, Ginner!</h1> | ||||||
|  | </body> | ||||||
|  | </html> | ||||||
|  | `)) | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	r := gin.Default() | ||||||
|  | 	r.Static("/assets", "./assets") | ||||||
|  | 	r.SetHTMLTemplate(html) | ||||||
|  | 
 | ||||||
|  | 	r.GET("/", func(c *gin.Context) { | ||||||
|  | 		if pusher := c.Writer.Pusher(); pusher != nil { | ||||||
|  | 			// use pusher.Push() to do server push | ||||||
|  | 			if err := pusher.Push("/assets/app.js", nil); err != nil { | ||||||
|  | 				log.Printf("Failed to push: %v", err) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		c.HTML(200, "https", gin.H{ | ||||||
|  | 			"status": "success", | ||||||
|  | 		}) | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	// Listen and Server in https://127.0.0.1:8080 | ||||||
|  | 	r.RunTLS(":8080", "./testdata/server.pem", "./testdata/server.key") | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								examples/http-pusher/testdata/ca.pem
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								examples/http-pusher/testdata/ca.pem
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | -----BEGIN CERTIFICATE----- | ||||||
|  | MIICSjCCAbOgAwIBAgIJAJHGGR4dGioHMA0GCSqGSIb3DQEBCwUAMFYxCzAJBgNV | ||||||
|  | BAYTAkFVMRMwEQYDVQQIEwpTb21lLVN0YXRlMSEwHwYDVQQKExhJbnRlcm5ldCBX | ||||||
|  | aWRnaXRzIFB0eSBMdGQxDzANBgNVBAMTBnRlc3RjYTAeFw0xNDExMTEyMjMxMjla | ||||||
|  | Fw0yNDExMDgyMjMxMjlaMFYxCzAJBgNVBAYTAkFVMRMwEQYDVQQIEwpTb21lLVN0 | ||||||
|  | YXRlMSEwHwYDVQQKExhJbnRlcm5ldCBXaWRnaXRzIFB0eSBMdGQxDzANBgNVBAMT | ||||||
|  | BnRlc3RjYTCBnzANBgkqhkiG9w0BAQEFAAOBjQAwgYkCgYEAwEDfBV5MYdlHVHJ7 | ||||||
|  | +L4nxrZy7mBfAVXpOc5vMYztssUI7mL2/iYujiIXM+weZYNTEpLdjyJdu7R5gGUu | ||||||
|  | g1jSVK/EPHfc74O7AyZU34PNIP4Sh33N+/A5YexrNgJlPY+E3GdVYi4ldWJjgkAd | ||||||
|  | Qah2PH5ACLrIIC6tRka9hcaBlIECAwEAAaMgMB4wDAYDVR0TBAUwAwEB/zAOBgNV | ||||||
|  | HQ8BAf8EBAMCAgQwDQYJKoZIhvcNAQELBQADgYEAHzC7jdYlzAVmddi/gdAeKPau | ||||||
|  | sPBG/C2HCWqHzpCUHcKuvMzDVkY/MP2o6JIW2DBbY64bO/FceExhjcykgaYtCH/m | ||||||
|  | oIU63+CFOTtR7otyQAWHqXa7q4SbCDlG7DyRFxqG0txPtGvy12lgldA2+RgcigQG | ||||||
|  | Dfcog5wrJytaQ6UA0wE= | ||||||
|  | -----END CERTIFICATE----- | ||||||
							
								
								
									
										16
									
								
								examples/http-pusher/testdata/server.key
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								examples/http-pusher/testdata/server.key
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | -----BEGIN PRIVATE KEY----- | ||||||
|  | MIICdQIBADANBgkqhkiG9w0BAQEFAASCAl8wggJbAgEAAoGBAOHDFScoLCVJpYDD | ||||||
|  | M4HYtIdV6Ake/sMNaaKdODjDMsux/4tDydlumN+fm+AjPEK5GHhGn1BgzkWF+slf | ||||||
|  | 3BxhrA/8dNsnunstVA7ZBgA/5qQxMfGAq4wHNVX77fBZOgp9VlSMVfyd9N8YwbBY | ||||||
|  | AckOeUQadTi2X1S6OgJXgQ0m3MWhAgMBAAECgYAn7qGnM2vbjJNBm0VZCkOkTIWm | ||||||
|  | V10okw7EPJrdL2mkre9NasghNXbE1y5zDshx5Nt3KsazKOxTT8d0Jwh/3KbaN+YY | ||||||
|  | tTCbKGW0pXDRBhwUHRcuRzScjli8Rih5UOCiZkhefUTcRb6xIhZJuQy71tjaSy0p | ||||||
|  | dHZRmYyBYO2YEQ8xoQJBAPrJPhMBkzmEYFtyIEqAxQ/o/A6E+E4w8i+KM7nQCK7q | ||||||
|  | K4JXzyXVAjLfyBZWHGM2uro/fjqPggGD6QH1qXCkI4MCQQDmdKeb2TrKRh5BY1LR | ||||||
|  | 81aJGKcJ2XbcDu6wMZK4oqWbTX2KiYn9GB0woM6nSr/Y6iy1u145YzYxEV/iMwff | ||||||
|  | DJULAkB8B2MnyzOg0pNFJqBJuH29bKCcHa8gHJzqXhNO5lAlEbMK95p/P2Wi+4Hd | ||||||
|  | aiEIAF1BF326QJcvYKmwSmrORp85AkAlSNxRJ50OWrfMZnBgzVjDx3xG6KsFQVk2 | ||||||
|  | ol6VhqL6dFgKUORFUWBvnKSyhjJxurlPEahV6oo6+A+mPhFY8eUvAkAZQyTdupP3 | ||||||
|  | XEFQKctGz+9+gKkemDp7LBBMEMBXrGTLPhpEfcjv/7KPdnFHYmhYeBTBnuVmTVWe | ||||||
|  | F98XJ7tIFfJq | ||||||
|  | -----END PRIVATE KEY----- | ||||||
							
								
								
									
										16
									
								
								examples/http-pusher/testdata/server.pem
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								examples/http-pusher/testdata/server.pem
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,16 @@ | |||||||
|  | -----BEGIN CERTIFICATE----- | ||||||
|  | MIICnDCCAgWgAwIBAgIBBzANBgkqhkiG9w0BAQsFADBWMQswCQYDVQQGEwJBVTET | ||||||
|  | MBEGA1UECBMKU29tZS1TdGF0ZTEhMB8GA1UEChMYSW50ZXJuZXQgV2lkZ2l0cyBQ | ||||||
|  | dHkgTHRkMQ8wDQYDVQQDEwZ0ZXN0Y2EwHhcNMTUxMTA0MDIyMDI0WhcNMjUxMTAx | ||||||
|  | MDIyMDI0WjBlMQswCQYDVQQGEwJVUzERMA8GA1UECBMISWxsaW5vaXMxEDAOBgNV | ||||||
|  | BAcTB0NoaWNhZ28xFTATBgNVBAoTDEV4YW1wbGUsIENvLjEaMBgGA1UEAxQRKi50 | ||||||
|  | ZXN0Lmdvb2dsZS5jb20wgZ8wDQYJKoZIhvcNAQEBBQADgY0AMIGJAoGBAOHDFSco | ||||||
|  | LCVJpYDDM4HYtIdV6Ake/sMNaaKdODjDMsux/4tDydlumN+fm+AjPEK5GHhGn1Bg | ||||||
|  | zkWF+slf3BxhrA/8dNsnunstVA7ZBgA/5qQxMfGAq4wHNVX77fBZOgp9VlSMVfyd | ||||||
|  | 9N8YwbBYAckOeUQadTi2X1S6OgJXgQ0m3MWhAgMBAAGjazBpMAkGA1UdEwQCMAAw | ||||||
|  | CwYDVR0PBAQDAgXgME8GA1UdEQRIMEaCECoudGVzdC5nb29nbGUuZnKCGHdhdGVy | ||||||
|  | em9vaS50ZXN0Lmdvb2dsZS5iZYISKi50ZXN0LnlvdXR1YmUuY29thwTAqAEDMA0G | ||||||
|  | CSqGSIb3DQEBCwUAA4GBAJFXVifQNub1LUP4JlnX5lXNlo8FxZ2a12AFQs+bzoJ6 | ||||||
|  | hM044EDjqyxUqSbVePK0ni3w1fHQB5rY9yYC5f8G7aqqTY1QOhoUk8ZTSTRpnkTh | ||||||
|  | y4jjdvTZeLDVBlueZUTDRmy2feY5aZIU18vFDK08dTG0A87pppuv1LNIR3loveU8 | ||||||
|  | -----END CERTIFICATE----- | ||||||
| @ -3,6 +3,7 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"html/template" | 	"html/template" | ||||||
| 	"log" | 	"log" | ||||||
|  | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| @ -27,7 +28,7 @@ func main() { | |||||||
| 	r.SetHTMLTemplate(html) | 	r.SetHTMLTemplate(html) | ||||||
| 
 | 
 | ||||||
| 	r.GET("/welcome", func(c *gin.Context) { | 	r.GET("/welcome", func(c *gin.Context) { | ||||||
| 		c.HTML(200, "https", gin.H{ | 		c.HTML(http.StatusOK, "https", gin.H{ | ||||||
| 			"status": "success", | 			"status": "success", | ||||||
| 		}) | 		}) | ||||||
| 	}) | 	}) | ||||||
|  | |||||||
							
								
								
									
										30
									
								
								examples/new_relic/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								examples/new_relic/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | |||||||
|  | The [New Relic Go Agent](https://github.com/newrelic/go-agent) provides a nice middleware for the stdlib handler signature.  | ||||||
|  | The following is an adaptation of that middleware for Gin. | ||||||
|  | 
 | ||||||
|  | ```golang | ||||||
|  | const ( | ||||||
|  | 	// NewRelicTxnKey is the key used to retrieve the NewRelic Transaction from the context | ||||||
|  | 	NewRelicTxnKey = "NewRelicTxnKey" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // NewRelicMonitoring is a middleware that starts a newrelic transaction, stores it in the context, then calls the next handler | ||||||
|  | func NewRelicMonitoring(app newrelic.Application) gin.HandlerFunc { | ||||||
|  | 	return func(ctx *gin.Context) { | ||||||
|  | 		txn := app.StartTransaction(ctx.Request.URL.Path, ctx.Writer, ctx.Request) | ||||||
|  | 		defer txn.End() | ||||||
|  | 		ctx.Set(NewRelicTxnKey, txn) | ||||||
|  | 		ctx.Next() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  | and in `main.go` or equivalent... | ||||||
|  | ```golang | ||||||
|  | router := gin.Default() | ||||||
|  | cfg := newrelic.NewConfig(os.Getenv("APP_NAME"), os.Getenv("NEW_RELIC_API_KEY")) | ||||||
|  | app, err := newrelic.NewApplication(cfg) | ||||||
|  | if err != nil { | ||||||
|  | 		log.Printf("failed to make new_relic app: %v", err) | ||||||
|  | } else { | ||||||
|  | 		router.Use(adapters.NewRelicMonitoring(app)) | ||||||
|  | } | ||||||
|  |  ``` | ||||||
							
								
								
									
										42
									
								
								examples/new_relic/main.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								examples/new_relic/main.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,42 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"log" | ||||||
|  | 	"net/http" | ||||||
|  | 	"os" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/newrelic/go-agent" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | const ( | ||||||
|  | 	// NewRelicTxnKey is the key used to retrieve the NewRelic Transaction from the context | ||||||
|  | 	NewRelicTxnKey = "NewRelicTxnKey" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // NewRelicMonitoring is a middleware that starts a newrelic transaction, stores it in the context, then calls the next handler | ||||||
|  | func NewRelicMonitoring(app newrelic.Application) gin.HandlerFunc { | ||||||
|  | 	return func(ctx *gin.Context) { | ||||||
|  | 		txn := app.StartTransaction(ctx.Request.URL.Path, ctx.Writer, ctx.Request) | ||||||
|  | 		defer txn.End() | ||||||
|  | 		ctx.Set(NewRelicTxnKey, txn) | ||||||
|  | 		ctx.Next() | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	router := gin.Default() | ||||||
|  | 
 | ||||||
|  | 	cfg := newrelic.NewConfig(os.Getenv("APP_NAME"), os.Getenv("NEW_RELIC_API_KEY")) | ||||||
|  | 	app, err := newrelic.NewApplication(cfg) | ||||||
|  | 	if err != nil { | ||||||
|  | 		log.Printf("failed to make new_relic app: %v", err) | ||||||
|  | 	} else { | ||||||
|  | 		router.Use(NewRelicMonitoring(app)) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	router.GET("/", func(c *gin.Context) { | ||||||
|  | 		c.String(http.StatusOK, "Hello World!\n") | ||||||
|  | 	}) | ||||||
|  | 	router.Run() | ||||||
|  | } | ||||||
| @ -13,16 +13,19 @@ func main() { | |||||||
| 	StartGin() | 	StartGin() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ConfigRuntime sets the number of operating system threads. | ||||||
| func ConfigRuntime() { | func ConfigRuntime() { | ||||||
| 	nuCPU := runtime.NumCPU() | 	nuCPU := runtime.NumCPU() | ||||||
| 	runtime.GOMAXPROCS(nuCPU) | 	runtime.GOMAXPROCS(nuCPU) | ||||||
| 	fmt.Printf("Running with %d CPUs\n", nuCPU) | 	fmt.Printf("Running with %d CPUs\n", nuCPU) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // StartWorkers start starsWorker by goroutine. | ||||||
| func StartWorkers() { | func StartWorkers() { | ||||||
| 	go statsWorker() | 	go statsWorker() | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // StartGin starts gin web server with setting router. | ||||||
| func StartGin() { | func StartGin() { | ||||||
| 	gin.SetMode(gin.ReleaseMode) | 	gin.SetMode(gin.ReleaseMode) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| <!DOCTYPE html> | <!doctype html> | ||||||
| <html lang="en"> | <html lang="en"> | ||||||
|     <head> |     <head> | ||||||
|         <meta charset="utf-8"> |         <meta charset="utf-8"> | ||||||
| @ -20,9 +20,9 @@ | |||||||
|         <!-- Latest compiled and minified JavaScript --> |         <!-- Latest compiled and minified JavaScript --> | ||||||
|         <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.4/js/bootstrap.min.js"></script> |         <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.4/js/bootstrap.min.js"></script> | ||||||
|         <!-- Primjs --> |         <!-- Primjs --> | ||||||
|         <link href="/static/prismjs.min.css" rel="stylesheet" /> |         <link href="/static/prismjs.min.css" rel="stylesheet"> | ||||||
| 
 | 
 | ||||||
|         <script type="text/javascript"> |         <script> | ||||||
|             $(document).ready(function() {  |             $(document).ready(function() {  | ||||||
|               StartRealtime({{.roomid}}, {{.timestamp}}); |               StartRealtime({{.roomid}}, {{.timestamp}}); | ||||||
|             }); |             }); | ||||||
| @ -49,7 +49,7 @@ | |||||||
|             <li><a href="http://www.w3.org/TR/2009/WD-eventsource-20091029/">W3 Standard</a></li> |             <li><a href="http://www.w3.org/TR/2009/WD-eventsource-20091029/">W3 Standard</a></li> | ||||||
|             <li><a href="http://caniuse.com/#feat=eventsource">Browser Support</a></li> |             <li><a href="http://caniuse.com/#feat=eventsource">Browser Support</a></li> | ||||||
|             <li><a href="http://gin-gonic.github.io/gin/">Gin Framework</a></li> |             <li><a href="http://gin-gonic.github.io/gin/">Gin Framework</a></li> | ||||||
|             <li><a href="https://github.com/gin-gonic/gin/tree/develop/examples/realtime-advanced">Github</a></li> |             <li><a href="https://github.com/gin-gonic/gin/tree/develop/examples/realtime-advanced">GitHub</a></li> | ||||||
|           </ul> |           </ul> | ||||||
|         </div><!-- /.nav-collapse --> |         </div><!-- /.nav-collapse --> | ||||||
|       </div><!-- /.container --> |       </div><!-- /.container --> | ||||||
| @ -59,7 +59,7 @@ | |||||||
|             <div class="container"> |             <div class="container"> | ||||||
|                 <h1>Server-Sent Events in Go</h1> |                 <h1>Server-Sent Events in Go</h1> | ||||||
|                 <p>Server-sent events (SSE) is a technology where a browser receives automatic updates from a server via HTTP connection. It is not websockets. <a href="http://www.html5rocks.com/en/tutorials/eventsource/basics/">Learn more.</a></p> |                 <p>Server-sent events (SSE) is a technology where a browser receives automatic updates from a server via HTTP connection. It is not websockets. <a href="http://www.html5rocks.com/en/tutorials/eventsource/basics/">Learn more.</a></p> | ||||||
|                 <p>The chat and the charts data is provided in realtime using the SSE implemention of <a href="https://github.com/gin-gonic/gin/blob/15b0c49da556d58a3d934b86e3aa552ff224026d/examples/realtime-chat/main.go#L23-L32">Gin Framework</a>.</p> |                 <p>The chat and the charts data is provided in realtime using the SSE implementation of <a href="https://github.com/gin-gonic/gin/blob/15b0c49da556d58a3d934b86e3aa552ff224026d/examples/realtime-chat/main.go#L23-L32">Gin Framework</a>.</p> | ||||||
|                 <div class="row"> |                 <div class="row"> | ||||||
|                     <div class="col-md-8"> |                     <div class="col-md-8"> | ||||||
|                         <div id="chat-scroll" style="overflow-y:scroll; overflow-x:scroll; height:290px"> |                         <div id="chat-scroll" style="overflow-y:scroll; overflow-x:scroll; height:290px"> | ||||||
| @ -79,19 +79,19 @@ | |||||||
|                                 <label class="sr-only" for="chat-message">Message</label> |                                 <label class="sr-only" for="chat-message">Message</label> | ||||||
|                                 <div class="input-group"> |                                 <div class="input-group"> | ||||||
|                                     <div class="input-group-addon">{{.nick}}</div> |                                     <div class="input-group-addon">{{.nick}}</div> | ||||||
|                                     <input type="text" name="message" id="chat-message" class="form-control" placeholder="a message" value="" /> |                                     <input type="text" name="message" id="chat-message" class="form-control" placeholder="a message" value=""> | ||||||
|                                 </div> |                                 </div> | ||||||
|                             </div> |                             </div> | ||||||
|                             <input type="submit" class="btn btn-primary" value="Send" /> |                             <input type="submit" class="btn btn-primary" value="Send"> | ||||||
|                         </form> |                         </form> | ||||||
|                         {{else}} |                         {{else}} | ||||||
|                         <form action="" method="get" class="form-inline"> |                         <form action="" method="get" class="form-inline"> | ||||||
|                             <legend>Join the SSE real-time chat</legend> |                             <legend>Join the SSE real-time chat</legend> | ||||||
|                             <div class="form-group"> |                             <div class="form-group"> | ||||||
|                                 <input value='' name="nick" id="nick" placeholder="Your Name" type="text" class="form-control" /> |                                 <input value='' name="nick" id="nick" placeholder="Your Name" type="text" class="form-control"> | ||||||
|                             </div> |                             </div> | ||||||
|                             <div class="form-group text-center"> |                             <div class="form-group text-center"> | ||||||
|                                 <input type="submit" class="btn btn-success btn-login-submit" value="Join" /> |                                 <input type="submit" class="btn btn-success btn-login-submit" value="Join"> | ||||||
|                             </div> |                             </div> | ||||||
|                         </form> |                         </form> | ||||||
|                         {{end}} |                         {{end}} | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"html" | 	"html" | ||||||
| 	"io" | 	"io" | ||||||
|  | 	"net/http" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| @ -21,12 +22,12 @@ func rateLimit(c *gin.Context) { | |||||||
| 			fmt.Println("ip blocked") | 			fmt.Println("ip blocked") | ||||||
| 		} | 		} | ||||||
| 		c.Abort() | 		c.Abort() | ||||||
| 		c.String(503, "you were automatically banned :)") | 		c.String(http.StatusServiceUnavailable, "you were automatically banned :)") | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func index(c *gin.Context) { | func index(c *gin.Context) { | ||||||
| 	c.Redirect(301, "/room/hn") | 	c.Redirect(http.StatusMovedPermanently, "/room/hn") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func roomGET(c *gin.Context) { | func roomGET(c *gin.Context) { | ||||||
| @ -38,7 +39,7 @@ func roomGET(c *gin.Context) { | |||||||
| 	if len(nick) > 13 { | 	if len(nick) > 13 { | ||||||
| 		nick = nick[0:12] + "..." | 		nick = nick[0:12] + "..." | ||||||
| 	} | 	} | ||||||
| 	c.HTML(200, "room_login.templ.html", gin.H{ | 	c.HTML(http.StatusOK, "room_login.templ.html", gin.H{ | ||||||
| 		"roomid":    roomid, | 		"roomid":    roomid, | ||||||
| 		"nick":      nick, | 		"nick":      nick, | ||||||
| 		"timestamp": time.Now().Unix(), | 		"timestamp": time.Now().Unix(), | ||||||
| @ -55,7 +56,7 @@ func roomPOST(c *gin.Context) { | |||||||
| 	validMessage := len(message) > 1 && len(message) < 200 | 	validMessage := len(message) > 1 && len(message) < 200 | ||||||
| 	validNick := len(nick) > 1 && len(nick) < 14 | 	validNick := len(nick) > 1 && len(nick) < 14 | ||||||
| 	if !validMessage || !validNick { | 	if !validMessage || !validNick { | ||||||
| 		c.JSON(400, gin.H{ | 		c.JSON(http.StatusBadRequest, gin.H{ | ||||||
| 			"status": "failed", | 			"status": "failed", | ||||||
| 			"error":  "the message or nickname is too long", | 			"error":  "the message or nickname is too long", | ||||||
| 		}) | 		}) | ||||||
| @ -68,7 +69,7 @@ func roomPOST(c *gin.Context) { | |||||||
| 	} | 	} | ||||||
| 	messages.Add("inbound", 1) | 	messages.Add("inbound", 1) | ||||||
| 	room(roomid).Submit(post) | 	room(roomid).Submit(post) | ||||||
| 	c.JSON(200, post) | 	c.JSON(http.StatusOK, post) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func streamRoom(c *gin.Context) { | func streamRoom(c *gin.Context) { | ||||||
|  | |||||||
| @ -50,6 +50,7 @@ func connectedUsers() uint64 { | |||||||
| 	return uint64(connected) | 	return uint64(connected) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Stats returns savedStats data. | ||||||
| func Stats() map[string]uint64 { | func Stats() map[string]uint64 { | ||||||
| 	mutexStats.RLock() | 	mutexStats.RLock() | ||||||
| 	defer mutexStats.RUnlock() | 	defer mutexStats.RUnlock() | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ import ( | |||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
| 	"math/rand" | 	"math/rand" | ||||||
|  | 	"net/http" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| ) | ) | ||||||
| @ -34,7 +35,7 @@ func stream(c *gin.Context) { | |||||||
| func roomGET(c *gin.Context) { | func roomGET(c *gin.Context) { | ||||||
| 	roomid := c.Param("roomid") | 	roomid := c.Param("roomid") | ||||||
| 	userid := fmt.Sprint(rand.Int31()) | 	userid := fmt.Sprint(rand.Int31()) | ||||||
| 	c.HTML(200, "chat_room", gin.H{ | 	c.HTML(http.StatusOK, "chat_room", gin.H{ | ||||||
| 		"roomid": roomid, | 		"roomid": roomid, | ||||||
| 		"userid": userid, | 		"userid": userid, | ||||||
| 	}) | 	}) | ||||||
| @ -46,7 +47,7 @@ func roomPOST(c *gin.Context) { | |||||||
| 	message := c.PostForm("message") | 	message := c.PostForm("message") | ||||||
| 	room(roomid).Submit(userid + ": " + message) | 	room(roomid).Submit(userid + ": " + message) | ||||||
| 
 | 
 | ||||||
| 	c.JSON(200, gin.H{ | 	c.JSON(http.StatusOK, gin.H{ | ||||||
| 		"status":  "success", | 		"status":  "success", | ||||||
| 		"message": message, | 		"message": message, | ||||||
| 	}) | 	}) | ||||||
|  | |||||||
| @ -6,7 +6,7 @@ var html = template.Must(template.New("chat_room").Parse(` | |||||||
| <html>  | <html>  | ||||||
| <head>  | <head>  | ||||||
|     <title>{{.roomid}}</title> |     <title>{{.roomid}}</title> | ||||||
|     <link rel="stylesheet" type="text/css" href="http://meyerweb.com/eric/tools/css/reset/reset.css"/> |     <link rel="stylesheet" type="text/css" href="http://meyerweb.com/eric/tools/css/reset/reset.css"> | ||||||
|     <script src="http://ajax.googleapis.com/ajax/libs/jquery/1.7/jquery.js"></script>  |     <script src="http://ajax.googleapis.com/ajax/libs/jquery/1.7/jquery.js"></script>  | ||||||
|     <script src="http://malsup.github.com/jquery.form.js"></script>  |     <script src="http://malsup.github.com/jquery.form.js"></script>  | ||||||
|     <script>  |     <script>  | ||||||
| @ -35,9 +35,9 @@ var html = template.Must(template.New("chat_room").Parse(` | |||||||
|     <h1>Welcome to {{.roomid}} room</h1> |     <h1>Welcome to {{.roomid}} room</h1> | ||||||
|     <div id="messages"></div> |     <div id="messages"></div> | ||||||
|     <form id="myForm" action="/room/{{.roomid}}" method="post">  |     <form id="myForm" action="/room/{{.roomid}}" method="post">  | ||||||
|     User: <input id="user_form" name="user" value="{{.userid}}"></input>  |     User: <input id="user_form" name="user" value="{{.userid}}"> | ||||||
|     Message: <input id="message_form" name="message"></input>  |     Message: <input id="message_form" name="message"> | ||||||
|     <input type="submit" value="Submit" />  |     <input type="submit" value="Submit">  | ||||||
|     </form> |     </form> | ||||||
| </body> | </body> | ||||||
| </html> | </html> | ||||||
|  | |||||||
							
								
								
									
										50
									
								
								examples/struct-lvl-validations/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								examples/struct-lvl-validations/README.md
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,50 @@ | |||||||
|  | ## Struct level validations | ||||||
|  | 
 | ||||||
|  | Validations can also be registered at the `struct` level when field level validations | ||||||
|  | don't make much sense. This can also be used to solve cross-field validation elegantly. | ||||||
|  | Additionally, it can be combined with tag validations. Struct Level validations run after | ||||||
|  | the structs tag validations. | ||||||
|  | 
 | ||||||
|  | ### Example requests | ||||||
|  | 
 | ||||||
|  | ```shell | ||||||
|  | # Validation errors are generated for struct tags as well as at the struct level | ||||||
|  | $ curl -s -X POST http://localhost:8085/user \ | ||||||
|  | 	-H 'content-type: application/json' \ | ||||||
|  | 	-d '{}' | jq | ||||||
|  | { | ||||||
|  |   "error": "Key: 'User.Email' Error:Field validation for 'Email' failed on the 'required' tag\nKey: 'User.FirstName' Error:Field validation for 'FirstName' failed on the 'fnameorlname' tag\nKey: 'User.LastName' Error:Field validation for 'LastName' failed on the 'fnameorlname' tag", | ||||||
|  |   "message": "User validation failed!" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | # Validation fails at the struct level because neither first name nor last name are present | ||||||
|  | $ curl -s -X POST http://localhost:8085/user \ | ||||||
|  |     -H 'content-type: application/json' \ | ||||||
|  | 	-d '{"email": "george@vandaley.com"}' | jq | ||||||
|  | { | ||||||
|  |   "error": "Key: 'User.FirstName' Error:Field validation for 'FirstName' failed on the 'fnameorlname' tag\nKey: 'User.LastName' Error:Field validation for 'LastName' failed on the 'fnameorlname' tag", | ||||||
|  |   "message": "User validation failed!" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | # No validation errors when either first name or last name is present | ||||||
|  | $ curl -X POST http://localhost:8085/user \ | ||||||
|  |     -H 'content-type: application/json' \ | ||||||
|  | 	-d '{"fname": "George", "email": "george@vandaley.com"}' | ||||||
|  | {"message":"User validation successful."} | ||||||
|  | 
 | ||||||
|  | $ curl -X POST http://localhost:8085/user \ | ||||||
|  |     -H 'content-type: application/json' \ | ||||||
|  | 	-d '{"lname": "Contanza", "email": "george@vandaley.com"}' | ||||||
|  | {"message":"User validation successful."} | ||||||
|  | 
 | ||||||
|  | $ curl -X POST http://localhost:8085/user \ | ||||||
|  |     -H 'content-type: application/json' \ | ||||||
|  | 	-d '{"fname": "George", "lname": "Costanza", "email": "george@vandaley.com"}' | ||||||
|  | {"message":"User validation successful."} | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | ### Useful links | ||||||
|  | 
 | ||||||
|  | - Validator docs - https://godoc.org/gopkg.in/go-playground/validator.v8#Validate.RegisterStructValidation | ||||||
|  | - Struct level example - https://github.com/go-playground/validator/blob/v8.18.2/examples/struct-level/struct_level.go | ||||||
|  | - Validator release notes - https://github.com/go-playground/validator/releases/tag/v8.7 | ||||||
							
								
								
									
										64
									
								
								examples/struct-lvl-validations/server.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								examples/struct-lvl-validations/server.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,64 @@ | |||||||
|  | package main | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 	"reflect" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin" | ||||||
|  | 	"github.com/gin-gonic/gin/binding" | ||||||
|  | 	validator "gopkg.in/go-playground/validator.v8" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // User contains user information. | ||||||
|  | type User struct { | ||||||
|  | 	FirstName string `json:"fname"` | ||||||
|  | 	LastName  string `json:"lname"` | ||||||
|  | 	Email     string `binding:"required,email"` | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // UserStructLevelValidation contains custom struct level validations that don't always | ||||||
|  | // make sense at the field validation level. For example, this function validates that either | ||||||
|  | // FirstName or LastName exist; could have done that with a custom field validation but then | ||||||
|  | // would have had to add it to both fields duplicating the logic + overhead, this way it's | ||||||
|  | // only validated once. | ||||||
|  | // | ||||||
|  | // NOTE: you may ask why wouldn't not just do this outside of validator. Doing this way | ||||||
|  | // hooks right into validator and you can combine with validation tags and still have a | ||||||
|  | // common error output format. | ||||||
|  | func UserStructLevelValidation(v *validator.Validate, structLevel *validator.StructLevel) { | ||||||
|  | 	user := structLevel.CurrentStruct.Interface().(User) | ||||||
|  | 
 | ||||||
|  | 	if len(user.FirstName) == 0 && len(user.LastName) == 0 { | ||||||
|  | 		structLevel.ReportError( | ||||||
|  | 			reflect.ValueOf(user.FirstName), "FirstName", "fname", "fnameorlname", | ||||||
|  | 		) | ||||||
|  | 		structLevel.ReportError( | ||||||
|  | 			reflect.ValueOf(user.LastName), "LastName", "lname", "fnameorlname", | ||||||
|  | 		) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	// plus can to more, even with different tag than "fnameorlname" | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func main() { | ||||||
|  | 	route := gin.Default() | ||||||
|  | 
 | ||||||
|  | 	if v, ok := binding.Validator.Engine().(*validator.Validate); ok { | ||||||
|  | 		v.RegisterStructValidation(UserStructLevelValidation, User{}) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	route.POST("/user", validateUser) | ||||||
|  | 	route.Run(":8085") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func validateUser(c *gin.Context) { | ||||||
|  | 	var u User | ||||||
|  | 	if err := c.ShouldBindJSON(&u); err == nil { | ||||||
|  | 		c.JSON(http.StatusOK, gin.H{"message": "User validation successful."}) | ||||||
|  | 	} else { | ||||||
|  | 		c.JSON(http.StatusBadRequest, gin.H{ | ||||||
|  | 			"message": "User validation failed!", | ||||||
|  | 			"error":   err.Error(), | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -20,7 +20,7 @@ func main() { | |||||||
| 	router.SetFuncMap(template.FuncMap{ | 	router.SetFuncMap(template.FuncMap{ | ||||||
| 		"formatAsDate": formatAsDate, | 		"formatAsDate": formatAsDate, | ||||||
| 	}) | 	}) | ||||||
| 	router.LoadHTMLFiles("../../fixtures/basic/raw.tmpl") | 	router.LoadHTMLFiles("../../testdata/template/raw.tmpl") | ||||||
| 
 | 
 | ||||||
| 	router.GET("/raw", func(c *gin.Context) { | 	router.GET("/raw", func(c *gin.Context) { | ||||||
| 		c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{ | 		c.HTML(http.StatusOK, "raw.tmpl", map[string]interface{}{ | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"path/filepath" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| ) | ) | ||||||
| @ -25,7 +26,8 @@ func main() { | |||||||
| 		files := form.File["files"] | 		files := form.File["files"] | ||||||
| 
 | 
 | ||||||
| 		for _, file := range files { | 		for _, file := range files { | ||||||
| 			if err := c.SaveUploadedFile(file, file.Filename); err != nil { | 			filename := filepath.Base(file.Filename) | ||||||
|  | 			if err := c.SaveUploadedFile(file, filename); err != nil { | ||||||
| 				c.String(http.StatusBadRequest, fmt.Sprintf("upload file err: %s", err.Error())) | 				c.String(http.StatusBadRequest, fmt.Sprintf("upload file err: %s", err.Error())) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ package main | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"path/filepath" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin" | 	"github.com/gin-gonic/gin" | ||||||
| ) | ) | ||||||
| @ -23,7 +24,8 @@ func main() { | |||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		if err := c.SaveUploadedFile(file, file.Filename); err != nil { | 		filename := filepath.Base(file.Filename) | ||||||
|  | 		if err := c.SaveUploadedFile(file, filename); err != nil { | ||||||
| 			c.String(http.StatusBadRequest, fmt.Sprintf("upload file err: %s", err.Error())) | 			c.String(http.StatusBadRequest, fmt.Sprintf("upload file err: %s", err.Error())) | ||||||
| 			return | 			return | ||||||
| 		} | 		} | ||||||
|  | |||||||
							
								
								
									
										144
									
								
								gin.go
									
									
									
									
									
								
							
							
						
						
									
										144
									
								
								gin.go
									
									
									
									
									
								
							| @ -5,6 +5,7 @@ | |||||||
| package gin | package gin | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
|  | 	"fmt" | ||||||
| 	"html/template" | 	"html/template" | ||||||
| 	"net" | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| @ -14,11 +15,7 @@ import ( | |||||||
| 	"github.com/gin-gonic/gin/render" | 	"github.com/gin-gonic/gin/render" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| const ( | const defaultMultipartMemory = 32 << 20 // 32 MB | ||||||
| 	// Version is Framework's version. |  | ||||||
| 	Version                = "v1.2" |  | ||||||
| 	defaultMultipartMemory = 32 << 20 // 32 MB |  | ||||||
| ) |  | ||||||
| 
 | 
 | ||||||
| var ( | var ( | ||||||
| 	default404Body   = []byte("404 page not found") | 	default404Body   = []byte("404 page not found") | ||||||
| @ -26,7 +23,10 @@ var ( | |||||||
| 	defaultAppEngine bool | 	defaultAppEngine bool | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // HandlerFunc defines the handler used by gin middleware as return value. | ||||||
| type HandlerFunc func(*Context) | type HandlerFunc func(*Context) | ||||||
|  | 
 | ||||||
|  | // HandlersChain defines a HandlerFunc array. | ||||||
| type HandlersChain []HandlerFunc | type HandlersChain []HandlerFunc | ||||||
| 
 | 
 | ||||||
| // Last returns the last handler in the chain. ie. the last handler is the main own. | // Last returns the last handler in the chain. ie. the last handler is the main own. | ||||||
| @ -37,12 +37,15 @@ func (c HandlersChain) Last() HandlerFunc { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // RouteInfo represents a request route's specification which contains method and path and its handler. | ||||||
| type RouteInfo struct { | type RouteInfo struct { | ||||||
| 	Method  string | 	Method      string | ||||||
| 	Path    string | 	Path        string | ||||||
| 	Handler string | 	Handler     string | ||||||
|  | 	HandlerFunc HandlerFunc | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // RoutesInfo defines a RouteInfo array. | ||||||
| type RoutesInfo []RouteInfo | type RoutesInfo []RouteInfo | ||||||
| 
 | 
 | ||||||
| // Engine is the framework's instance, it contains the muxer, middleware and configuration settings. | // Engine is the framework's instance, it contains the muxer, middleware and configuration settings. | ||||||
| @ -155,6 +158,7 @@ func (engine *Engine) allocateContext() *Context { | |||||||
| 	return &Context{engine: engine} | 	return &Context{engine: engine} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Delims sets template left and right delims and returns a Engine instance. | ||||||
| func (engine *Engine) Delims(left, right string) *Engine { | func (engine *Engine) Delims(left, right string) *Engine { | ||||||
| 	engine.delims = render.Delims{Left: left, Right: right} | 	engine.delims = render.Delims{Left: left, Right: right} | ||||||
| 	return engine | 	return engine | ||||||
| @ -171,14 +175,14 @@ func (engine *Engine) SecureJsonPrefix(prefix string) *Engine { | |||||||
| func (engine *Engine) LoadHTMLGlob(pattern string) { | func (engine *Engine) LoadHTMLGlob(pattern string) { | ||||||
| 	left := engine.delims.Left | 	left := engine.delims.Left | ||||||
| 	right := engine.delims.Right | 	right := engine.delims.Right | ||||||
|  | 	templ := template.Must(template.New("").Delims(left, right).Funcs(engine.FuncMap).ParseGlob(pattern)) | ||||||
| 
 | 
 | ||||||
| 	if IsDebugging() { | 	if IsDebugging() { | ||||||
| 		debugPrintLoadTemplate(template.Must(template.New("").Delims(left, right).Funcs(engine.FuncMap).ParseGlob(pattern))) | 		debugPrintLoadTemplate(templ) | ||||||
| 		engine.HTMLRender = render.HTMLDebug{Glob: pattern, FuncMap: engine.FuncMap, Delims: engine.delims} | 		engine.HTMLRender = render.HTMLDebug{Glob: pattern, FuncMap: engine.FuncMap, Delims: engine.delims} | ||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	templ := template.Must(template.New("").Delims(left, right).Funcs(engine.FuncMap).ParseGlob(pattern)) |  | ||||||
| 	engine.SetHTMLTemplate(templ) | 	engine.SetHTMLTemplate(templ) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -264,10 +268,12 @@ func (engine *Engine) Routes() (routes RoutesInfo) { | |||||||
| func iterate(path, method string, routes RoutesInfo, root *node) RoutesInfo { | func iterate(path, method string, routes RoutesInfo, root *node) RoutesInfo { | ||||||
| 	path += root.path | 	path += root.path | ||||||
| 	if len(root.handlers) > 0 { | 	if len(root.handlers) > 0 { | ||||||
|  | 		handlerFunc := root.handlers.Last() | ||||||
| 		routes = append(routes, RouteInfo{ | 		routes = append(routes, RouteInfo{ | ||||||
| 			Method:  method, | 			Method:      method, | ||||||
| 			Path:    path, | 			Path:        path, | ||||||
| 			Handler: nameOfFunction(root.handlers.Last()), | 			Handler:     nameOfFunction(handlerFunc), | ||||||
|  | 			HandlerFunc: handlerFunc, | ||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
| 	for _, child := range root.children { | 	for _, child := range root.children { | ||||||
| @ -312,6 +318,24 @@ func (engine *Engine) RunUnix(file string) (err error) { | |||||||
| 		return | 		return | ||||||
| 	} | 	} | ||||||
| 	defer listener.Close() | 	defer listener.Close() | ||||||
|  | 	os.Chmod(file, 0777) | ||||||
|  | 	err = http.Serve(listener, engine) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // RunFd attaches the router to a http.Server and starts listening and serving HTTP requests | ||||||
|  | // through the specified file descriptor. | ||||||
|  | // Note: this method will block the calling goroutine indefinitely unless an error happens. | ||||||
|  | func (engine *Engine) RunFd(fd int) (err error) { | ||||||
|  | 	debugPrint("Listening and serving HTTP on fd@%d", fd) | ||||||
|  | 	defer func() { debugPrintError(err) }() | ||||||
|  | 
 | ||||||
|  | 	f := os.NewFile(uintptr(fd), fmt.Sprintf("fd@%d", fd)) | ||||||
|  | 	listener, err := net.FileListener(f) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	defer listener.Close() | ||||||
| 	err = http.Serve(listener, engine) | 	err = http.Serve(listener, engine) | ||||||
| 	return | 	return | ||||||
| } | } | ||||||
| @ -329,12 +353,14 @@ func (engine *Engine) ServeHTTP(w http.ResponseWriter, req *http.Request) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // HandleContext re-enter a context that has been rewritten. | // HandleContext re-enter a context that has been rewritten. | ||||||
| // This can be done by setting c.Request.Path to your new target. | // This can be done by setting c.Request.URL.Path to your new target. | ||||||
| // Disclaimer: You can loop yourself to death with this, use wisely. | // Disclaimer: You can loop yourself to death with this, use wisely. | ||||||
| func (engine *Engine) HandleContext(c *Context) { | func (engine *Engine) HandleContext(c *Context) { | ||||||
|  | 	oldIndexValue := c.index | ||||||
| 	c.reset() | 	c.reset() | ||||||
| 	engine.handleHTTPRequest(c) | 	engine.handleHTTPRequest(c) | ||||||
| 	engine.pool.Put(c) | 
 | ||||||
|  | 	c.index = oldIndexValue | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func (engine *Engine) handleHTTPRequest(c *Context) { | func (engine *Engine) handleHTTPRequest(c *Context) { | ||||||
| @ -349,43 +375,45 @@ func (engine *Engine) handleHTTPRequest(c *Context) { | |||||||
| 	// Find root of the tree for the given HTTP method | 	// Find root of the tree for the given HTTP method | ||||||
| 	t := engine.trees | 	t := engine.trees | ||||||
| 	for i, tl := 0, len(t); i < tl; i++ { | 	for i, tl := 0, len(t); i < tl; i++ { | ||||||
| 		if t[i].method == httpMethod { | 		if t[i].method != httpMethod { | ||||||
| 			root := t[i].root | 			continue | ||||||
| 			// Find route in tree | 		} | ||||||
| 			handlers, params, tsr := root.getValue(path, c.Params, unescape) | 		root := t[i].root | ||||||
| 			if handlers != nil { | 		// Find route in tree | ||||||
| 				c.handlers = handlers | 		handlers, params, tsr := root.getValue(path, c.Params, unescape) | ||||||
| 				c.Params = params | 		if handlers != nil { | ||||||
| 				c.Next() | 			c.handlers = handlers | ||||||
| 				c.writermem.WriteHeaderNow() | 			c.Params = params | ||||||
|  | 			c.Next() | ||||||
|  | 			c.writermem.WriteHeaderNow() | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		if httpMethod != "CONNECT" && path != "/" { | ||||||
|  | 			if tsr && engine.RedirectTrailingSlash { | ||||||
|  | 				redirectTrailingSlash(c) | ||||||
| 				return | 				return | ||||||
| 			} | 			} | ||||||
| 			if httpMethod != "CONNECT" && path != "/" { | 			if engine.RedirectFixedPath && redirectFixedPath(c, root, engine.RedirectFixedPath) { | ||||||
| 				if tsr && engine.RedirectTrailingSlash { | 				return | ||||||
| 					redirectTrailingSlash(c) |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 				if engine.RedirectFixedPath && redirectFixedPath(c, root, engine.RedirectFixedPath) { |  | ||||||
| 					return |  | ||||||
| 				} |  | ||||||
| 			} | 			} | ||||||
| 			break |  | ||||||
| 		} | 		} | ||||||
|  | 		break | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	if engine.HandleMethodNotAllowed { | 	if engine.HandleMethodNotAllowed { | ||||||
| 		for _, tree := range engine.trees { | 		for _, tree := range engine.trees { | ||||||
| 			if tree.method != httpMethod { | 			if tree.method == httpMethod { | ||||||
| 				if handlers, _, _ := tree.root.getValue(path, nil, unescape); handlers != nil { | 				continue | ||||||
| 					c.handlers = engine.allNoMethod | 			} | ||||||
| 					serveError(c, 405, default405Body) | 			if handlers, _, _ := tree.root.getValue(path, nil, unescape); handlers != nil { | ||||||
| 					return | 				c.handlers = engine.allNoMethod | ||||||
| 				} | 				serveError(c, http.StatusMethodNotAllowed, default405Body) | ||||||
|  | 				return | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 	c.handlers = engine.allNoRoute | 	c.handlers = engine.allNoRoute | ||||||
| 	serveError(c, 404, default404Body) | 	serveError(c, http.StatusNotFound, default404Body) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var mimePlain = []string{MIMEPlain} | var mimePlain = []string{MIMEPlain} | ||||||
| @ -393,28 +421,32 @@ var mimePlain = []string{MIMEPlain} | |||||||
| func serveError(c *Context, code int, defaultMessage []byte) { | func serveError(c *Context, code int, defaultMessage []byte) { | ||||||
| 	c.writermem.status = code | 	c.writermem.status = code | ||||||
| 	c.Next() | 	c.Next() | ||||||
| 	if !c.writermem.Written() { | 	if c.writermem.Written() { | ||||||
| 		if c.writermem.Status() == code { | 		return | ||||||
| 			c.writermem.Header()["Content-Type"] = mimePlain |  | ||||||
| 			c.Writer.Write(defaultMessage) |  | ||||||
| 		} else { |  | ||||||
| 			c.writermem.WriteHeaderNow() |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
|  | 	if c.writermem.Status() == code { | ||||||
|  | 		c.writermem.Header()["Content-Type"] = mimePlain | ||||||
|  | 		_, err := c.Writer.Write(defaultMessage) | ||||||
|  | 		if err != nil { | ||||||
|  | 			debugPrint("cannot write message to writer during serve error: %v", err) | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	c.writermem.WriteHeaderNow() | ||||||
|  | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func redirectTrailingSlash(c *Context) { | func redirectTrailingSlash(c *Context) { | ||||||
| 	req := c.Request | 	req := c.Request | ||||||
| 	path := req.URL.Path | 	path := req.URL.Path | ||||||
| 	code := 301 // Permanent redirect, request with GET method | 	code := http.StatusMovedPermanently // Permanent redirect, request with GET method | ||||||
| 	if req.Method != "GET" { | 	if req.Method != "GET" { | ||||||
| 		code = 307 | 		code = http.StatusTemporaryRedirect | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
|  | 	req.URL.Path = path + "/" | ||||||
| 	if length := len(path); length > 1 && path[length-1] == '/' { | 	if length := len(path); length > 1 && path[length-1] == '/' { | ||||||
| 		req.URL.Path = path[:length-1] | 		req.URL.Path = path[:length-1] | ||||||
| 	} else { |  | ||||||
| 		req.URL.Path = path + "/" |  | ||||||
| 	} | 	} | ||||||
| 	debugPrint("redirecting request %d: %s --> %s", code, path, req.URL.String()) | 	debugPrint("redirecting request %d: %s --> %s", code, path, req.URL.String()) | ||||||
| 	http.Redirect(c.Writer, req, req.URL.String(), code) | 	http.Redirect(c.Writer, req, req.URL.String(), code) | ||||||
| @ -425,14 +457,10 @@ func redirectFixedPath(c *Context, root *node, trailingSlash bool) bool { | |||||||
| 	req := c.Request | 	req := c.Request | ||||||
| 	path := req.URL.Path | 	path := req.URL.Path | ||||||
| 
 | 
 | ||||||
| 	fixedPath, found := root.findCaseInsensitivePath( | 	if fixedPath, ok := root.findCaseInsensitivePath(cleanPath(path), trailingSlash); ok { | ||||||
| 		cleanPath(path), | 		code := http.StatusMovedPermanently // Permanent redirect, request with GET method | ||||||
| 		trailingSlash, |  | ||||||
| 	) |  | ||||||
| 	if found { |  | ||||||
| 		code := 301 // Permanent redirect, request with GET method |  | ||||||
| 		if req.Method != "GET" { | 		if req.Method != "GET" { | ||||||
| 			code = 307 | 			code = http.StatusTemporaryRedirect | ||||||
| 		} | 		} | ||||||
| 		req.URL.Path = string(fixedPath) | 		req.URL.Path = string(fixedPath) | ||||||
| 		debugPrint("redirecting request %d: %s --> %s", code, path, req.URL.String()) | 		debugPrint("redirecting request %d: %s --> %s", code, path, req.URL.String()) | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								ginS/gins.go
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								ginS/gins.go
									
									
									
									
									
								
							| @ -22,14 +22,17 @@ func engine() *gin.Engine { | |||||||
| 	return internalEngine | 	return internalEngine | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // LoadHTMLGlob is a wrapper for Engine.LoadHTMLGlob. | ||||||
| func LoadHTMLGlob(pattern string) { | func LoadHTMLGlob(pattern string) { | ||||||
| 	engine().LoadHTMLGlob(pattern) | 	engine().LoadHTMLGlob(pattern) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // LoadHTMLFiles is a wrapper for Engine.LoadHTMLFiles. | ||||||
| func LoadHTMLFiles(files ...string) { | func LoadHTMLFiles(files ...string) { | ||||||
| 	engine().LoadHTMLFiles(files...) | 	engine().LoadHTMLFiles(files...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // SetHTMLTemplate is a wrapper for Engine.SetHTMLTemplate. | ||||||
| func SetHTMLTemplate(templ *template.Template) { | func SetHTMLTemplate(templ *template.Template) { | ||||||
| 	engine().SetHTMLTemplate(templ) | 	engine().SetHTMLTemplate(templ) | ||||||
| } | } | ||||||
| @ -39,17 +42,18 @@ func NoRoute(handlers ...gin.HandlerFunc) { | |||||||
| 	engine().NoRoute(handlers...) | 	engine().NoRoute(handlers...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // NoMethod sets the handlers called when... TODO | // NoMethod is a wrapper for Engine.NoMethod. | ||||||
| func NoMethod(handlers ...gin.HandlerFunc) { | func NoMethod(handlers ...gin.HandlerFunc) { | ||||||
| 	engine().NoMethod(handlers...) | 	engine().NoMethod(handlers...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // Group creates a new router group. You should add all the routes that have common middlwares or the same path prefix. | // Group creates a new router group. You should add all the routes that have common middlewares or the same path prefix. | ||||||
| // For example, all the routes that use a common middlware for authorization could be grouped. | // For example, all the routes that use a common middleware for authorization could be grouped. | ||||||
| func Group(relativePath string, handlers ...gin.HandlerFunc) *gin.RouterGroup { | func Group(relativePath string, handlers ...gin.HandlerFunc) *gin.RouterGroup { | ||||||
| 	return engine().Group(relativePath, handlers...) | 	return engine().Group(relativePath, handlers...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Handle is a wrapper for Engine.Handle. | ||||||
| func Handle(httpMethod, relativePath string, handlers ...gin.HandlerFunc) gin.IRoutes { | func Handle(httpMethod, relativePath string, handlers ...gin.HandlerFunc) gin.IRoutes { | ||||||
| 	return engine().Handle(httpMethod, relativePath, handlers...) | 	return engine().Handle(httpMethod, relativePath, handlers...) | ||||||
| } | } | ||||||
| @ -89,10 +93,12 @@ func HEAD(relativePath string, handlers ...gin.HandlerFunc) gin.IRoutes { | |||||||
| 	return engine().HEAD(relativePath, handlers...) | 	return engine().HEAD(relativePath, handlers...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Any is a wrapper for Engine.Any. | ||||||
| func Any(relativePath string, handlers ...gin.HandlerFunc) gin.IRoutes { | func Any(relativePath string, handlers ...gin.HandlerFunc) gin.IRoutes { | ||||||
| 	return engine().Any(relativePath, handlers...) | 	return engine().Any(relativePath, handlers...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // StaticFile is a wrapper for Engine.StaticFile. | ||||||
| func StaticFile(relativePath, filepath string) gin.IRoutes { | func StaticFile(relativePath, filepath string) gin.IRoutes { | ||||||
| 	return engine().StaticFile(relativePath, filepath) | 	return engine().StaticFile(relativePath, filepath) | ||||||
| } | } | ||||||
| @ -107,6 +113,7 @@ func Static(relativePath, root string) gin.IRoutes { | |||||||
| 	return engine().Static(relativePath, root) | 	return engine().Static(relativePath, root) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // StaticFS is a wrapper for Engine.StaticFS. | ||||||
| func StaticFS(relativePath string, fs http.FileSystem) gin.IRoutes { | func StaticFS(relativePath string, fs http.FileSystem) gin.IRoutes { | ||||||
| 	return engine().StaticFS(relativePath, fs) | 	return engine().StaticFS(relativePath, fs) | ||||||
| } | } | ||||||
| @ -120,21 +127,21 @@ func Use(middlewares ...gin.HandlerFunc) gin.IRoutes { | |||||||
| 
 | 
 | ||||||
| // Run : The router is attached to a http.Server and starts listening and serving HTTP requests. | // Run : The router is attached to a http.Server and starts listening and serving HTTP requests. | ||||||
| // It is a shortcut for http.ListenAndServe(addr, router) | // It is a shortcut for http.ListenAndServe(addr, router) | ||||||
| // Note: this method will block the calling goroutine undefinitelly unless an error happens. | // Note: this method will block the calling goroutine indefinitely unless an error happens. | ||||||
| func Run(addr ...string) (err error) { | func Run(addr ...string) (err error) { | ||||||
| 	return engine().Run(addr...) | 	return engine().Run(addr...) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // RunTLS : The router is attached to a http.Server and starts listening and serving HTTPS requests. | // RunTLS : The router is attached to a http.Server and starts listening and serving HTTPS requests. | ||||||
| // It is a shortcut for http.ListenAndServeTLS(addr, certFile, keyFile, router) | // It is a shortcut for http.ListenAndServeTLS(addr, certFile, keyFile, router) | ||||||
| // Note: this method will block the calling goroutine undefinitelly unless an error happens. | // Note: this method will block the calling goroutine indefinitely unless an error happens. | ||||||
| func RunTLS(addr string, certFile string, keyFile string) (err error) { | func RunTLS(addr, certFile, keyFile string) (err error) { | ||||||
| 	return engine().RunTLS(addr, certFile, keyFile) | 	return engine().RunTLS(addr, certFile, keyFile) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // RunUnix : The router is attached to a http.Server and starts listening and serving HTTP requests | // RunUnix : The router is attached to a http.Server and starts listening and serving HTTP requests | ||||||
| // through the specified unix socket (ie. a file) | // through the specified unix socket (ie. a file) | ||||||
| // Note: this method will block the calling goroutine undefinitelly unless an error happens. | // Note: this method will block the calling goroutine indefinitely unless an error happens. | ||||||
| func RunUnix(file string) (err error) { | func RunUnix(file string) (err error) { | ||||||
| 	return engine().RunUnix(file) | 	return engine().RunUnix(file) | ||||||
| } | } | ||||||
|  | |||||||
| @ -6,12 +6,14 @@ package gin | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bufio" | 	"bufio" | ||||||
|  | 	"crypto/tls" | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net" | 	"net" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 	"net/http/httptest" | 	"net/http/httptest" | ||||||
| 	"os" | 	"os" | ||||||
|  | 	"sync" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| @ -19,7 +21,14 @@ import ( | |||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func testRequest(t *testing.T, url string) { | func testRequest(t *testing.T, url string) { | ||||||
| 	resp, err := http.Get(url) | 	tr := &http.Transport{ | ||||||
|  | 		TLSClientConfig: &tls.Config{ | ||||||
|  | 			InsecureSkipVerify: true, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  | 	client := &http.Client{Transport: tr} | ||||||
|  | 
 | ||||||
|  | 	resp, err := client.Get(url) | ||||||
| 	assert.NoError(t, err) | 	assert.NoError(t, err) | ||||||
| 	defer resp.Body.Close() | 	defer resp.Body.Close() | ||||||
| 
 | 
 | ||||||
| @ -44,6 +53,22 @@ func TestRunEmpty(t *testing.T) { | |||||||
| 	testRequest(t, "http://localhost:8080/example") | 	testRequest(t, "http://localhost:8080/example") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestRunTLS(t *testing.T) { | ||||||
|  | 	router := New() | ||||||
|  | 	go func() { | ||||||
|  | 		router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | ||||||
|  | 
 | ||||||
|  | 		assert.NoError(t, router.RunTLS(":8443", "./testdata/certificate/cert.pem", "./testdata/certificate/key.pem")) | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	// have to wait for the goroutine to start and run the server | ||||||
|  | 	// otherwise the main thread will complete | ||||||
|  | 	time.Sleep(5 * time.Millisecond) | ||||||
|  | 
 | ||||||
|  | 	assert.Error(t, router.RunTLS(":8443", "./testdata/certificate/cert.pem", "./testdata/certificate/key.pem")) | ||||||
|  | 	testRequest(t, "https://localhost:8443/example") | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestRunEmptyWithEnv(t *testing.T) { | func TestRunEmptyWithEnv(t *testing.T) { | ||||||
| 	os.Setenv("PORT", "3123") | 	os.Setenv("PORT", "3123") | ||||||
| 	router := New() | 	router := New() | ||||||
| @ -62,7 +87,7 @@ func TestRunEmptyWithEnv(t *testing.T) { | |||||||
| func TestRunTooMuchParams(t *testing.T) { | func TestRunTooMuchParams(t *testing.T) { | ||||||
| 	router := New() | 	router := New() | ||||||
| 	assert.Panics(t, func() { | 	assert.Panics(t, func() { | ||||||
| 		router.Run("2", "2") | 		assert.NoError(t, router.Run("2", "2")) | ||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -109,6 +134,42 @@ func TestBadUnixSocket(t *testing.T) { | |||||||
| 	assert.Error(t, router.RunUnix("#/tmp/unix_unit_test")) | 	assert.Error(t, router.RunUnix("#/tmp/unix_unit_test")) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestFileDescriptor(t *testing.T) { | ||||||
|  | 	router := New() | ||||||
|  | 
 | ||||||
|  | 	addr, err := net.ResolveTCPAddr("tcp", "localhost:0") | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	listener, err := net.ListenTCP("tcp", addr) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	socketFile, err := listener.File() | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 	go func() { | ||||||
|  | 		router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | ||||||
|  | 		assert.NoError(t, router.RunFd(int(socketFile.Fd()))) | ||||||
|  | 	}() | ||||||
|  | 	// have to wait for the goroutine to start and run the server | ||||||
|  | 	// otherwise the main thread will complete | ||||||
|  | 	time.Sleep(5 * time.Millisecond) | ||||||
|  | 
 | ||||||
|  | 	c, err := net.Dial("tcp", listener.Addr().String()) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 	fmt.Fprintf(c, "GET /example HTTP/1.0\r\n\r\n") | ||||||
|  | 	scanner := bufio.NewScanner(c) | ||||||
|  | 	var response string | ||||||
|  | 	for scanner.Scan() { | ||||||
|  | 		response += scanner.Text() | ||||||
|  | 	} | ||||||
|  | 	assert.Contains(t, response, "HTTP/1.0 200", "should get a 200") | ||||||
|  | 	assert.Contains(t, response, "it worked", "resp body should match") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBadFileDescriptor(t *testing.T) { | ||||||
|  | 	router := New() | ||||||
|  | 	assert.Error(t, router.RunFd(0)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestWithHttptestWithAutoSelectedPort(t *testing.T) { | func TestWithHttptestWithAutoSelectedPort(t *testing.T) { | ||||||
| 	router := New() | 	router := New() | ||||||
| 	router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | 	router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | ||||||
| @ -119,6 +180,26 @@ func TestWithHttptestWithAutoSelectedPort(t *testing.T) { | |||||||
| 	testRequest(t, ts.URL+"/example") | 	testRequest(t, ts.URL+"/example") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestConcurrentHandleContext(t *testing.T) { | ||||||
|  | 	router := New() | ||||||
|  | 	router.GET("/", func(c *Context) { | ||||||
|  | 		c.Request.URL.Path = "/example" | ||||||
|  | 		router.HandleContext(c) | ||||||
|  | 	}) | ||||||
|  | 	router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | ||||||
|  | 
 | ||||||
|  | 	var wg sync.WaitGroup | ||||||
|  | 	iterations := 200 | ||||||
|  | 	wg.Add(iterations) | ||||||
|  | 	for i := 0; i < iterations; i++ { | ||||||
|  | 		go func() { | ||||||
|  | 			testGetRequestHandler(t, router, "/") | ||||||
|  | 			wg.Done() | ||||||
|  | 		}() | ||||||
|  | 	} | ||||||
|  | 	wg.Wait() | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // func TestWithHttptestWithSpecifiedPort(t *testing.T) { | // func TestWithHttptestWithSpecifiedPort(t *testing.T) { | ||||||
| // 	router := New() | // 	router := New() | ||||||
| // 	router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | // 	router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") }) | ||||||
| @ -133,3 +214,14 @@ func TestWithHttptestWithAutoSelectedPort(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| // 	testRequest(t, "http://localhost:8033/example") | // 	testRequest(t, "http://localhost:8033/example") | ||||||
| // } | // } | ||||||
|  | 
 | ||||||
|  | func testGetRequestHandler(t *testing.T, h http.Handler, url string) { | ||||||
|  | 	req, err := http.NewRequest("GET", url, nil) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 
 | ||||||
|  | 	w := httptest.NewRecorder() | ||||||
|  | 	h.ServeHTTP(w, req) | ||||||
|  | 
 | ||||||
|  | 	assert.Equal(t, "it worked", w.Body.String(), "resp body should match") | ||||||
|  | 	assert.Equal(t, 200, w.Code, "should get a 200") | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										311
									
								
								gin_test.go
									
									
									
									
									
								
							
							
						
						
									
										311
									
								
								gin_test.go
									
									
									
									
									
								
							| @ -10,7 +10,10 @@ import ( | |||||||
| 	"html/template" | 	"html/template" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"net/http" | 	"net/http" | ||||||
|  | 	"net/http/httptest" | ||||||
| 	"reflect" | 	"reflect" | ||||||
|  | 	"strconv" | ||||||
|  | 	"sync/atomic" | ||||||
| 	"testing" | 	"testing" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| @ -22,15 +25,18 @@ func formatAsDate(t time.Time) string { | |||||||
| 	return fmt.Sprintf("%d/%02d/%02d", year, month, day) | 	return fmt.Sprintf("%d/%02d/%02d", year, month, day) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func setupHTMLFiles(t *testing.T, mode string, tls bool) func() { | func setupHTMLFiles(t *testing.T, mode string, tls bool, loadMethod func(*Engine)) *httptest.Server { | ||||||
| 	go func() { | 	SetMode(mode) | ||||||
| 		SetMode(mode) | 	defer SetMode(TestMode) | ||||||
| 		router := New() | 
 | ||||||
|  | 	var router *Engine | ||||||
|  | 	captureOutput(t, func() { | ||||||
|  | 		router = New() | ||||||
| 		router.Delims("{[{", "}]}") | 		router.Delims("{[{", "}]}") | ||||||
| 		router.SetFuncMap(template.FuncMap{ | 		router.SetFuncMap(template.FuncMap{ | ||||||
| 			"formatAsDate": formatAsDate, | 			"formatAsDate": formatAsDate, | ||||||
| 		}) | 		}) | ||||||
| 		router.LoadHTMLFiles("./fixtures/basic/hello.tmpl", "./fixtures/basic/raw.tmpl") | 		loadMethod(router) | ||||||
| 		router.GET("/test", func(c *Context) { | 		router.GET("/test", func(c *Context) { | ||||||
| 			c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"}) | 			c.HTML(http.StatusOK, "hello.tmpl", map[string]string{"name": "world"}) | ||||||
| 		}) | 		}) | ||||||
| @ -39,88 +45,90 @@ func setupHTMLFiles(t *testing.T, mode string, tls bool) func() { | |||||||
| 				"now": time.Date(2017, 07, 01, 0, 0, 0, 0, time.UTC), | 				"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") | 	var ts *httptest.Server | ||||||
| 		} else { | 
 | ||||||
| 			router.Run(":8888") | 	if tls { | ||||||
| 		} | 		ts = httptest.NewTLSServer(router) | ||||||
| 	}() | 	} else { | ||||||
| 	t.Log("waiting 1 second for server startup") | 		ts = httptest.NewServer(router) | ||||||
| 	time.Sleep(1 * time.Second) | 	} | ||||||
| 	return func() {} | 
 | ||||||
|  | 	return ts | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func setupHTMLGlob(t *testing.T, mode string, tls bool) func() { | func TestLoadHTMLGlobDebugMode(t *testing.T) { | ||||||
| 	go func() { | 	ts := setupHTMLFiles( | ||||||
| 		SetMode(mode) | 		t, | ||||||
| 		router := New() | 		DebugMode, | ||||||
| 		router.Delims("{[{", "}]}") | 		false, | ||||||
| 		router.SetFuncMap(template.FuncMap{ | 		func(router *Engine) { | ||||||
| 			"formatAsDate": formatAsDate, | 			router.LoadHTMLGlob("./testdata/template/*") | ||||||
| 		}) | 		}, | ||||||
| 		router.LoadHTMLGlob("./fixtures/basic/*") | 	) | ||||||
| 		router.GET("/test", func(c *Context) { | 	defer ts.Close() | ||||||
| 			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() {} |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLGlob(t *testing.T) { | 	res, err := http.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	td := setupHTMLGlob(t, DebugMode, false) |  | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLGlob2(t *testing.T) { | func TestLoadHTMLGlobTestMode(t *testing.T) { | ||||||
| 	td := setupHTMLGlob(t, TestMode, false) | 	ts := setupHTMLFiles( | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") | 		t, | ||||||
|  | 		TestMode, | ||||||
|  | 		false, | ||||||
|  | 		func(router *Engine) { | ||||||
|  | 			router.LoadHTMLGlob("./testdata/template/*") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | 	defer ts.Close() | ||||||
|  | 
 | ||||||
|  | 	res, err := http.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLGlob3(t *testing.T) { | func TestLoadHTMLGlobReleaseMode(t *testing.T) { | ||||||
| 	td := setupHTMLGlob(t, ReleaseMode, false) | 	ts := setupHTMLFiles( | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") | 		t, | ||||||
|  | 		ReleaseMode, | ||||||
|  | 		false, | ||||||
|  | 		func(router *Engine) { | ||||||
|  | 			router.LoadHTMLGlob("./testdata/template/*") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | 	defer ts.Close() | ||||||
|  | 
 | ||||||
|  | 	res, err := http.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLGlobUsingTLS(t *testing.T) { | func TestLoadHTMLGlobUsingTLS(t *testing.T) { | ||||||
| 	td := setupHTMLGlob(t, DebugMode, true) | 	ts := setupHTMLFiles( | ||||||
|  | 		t, | ||||||
|  | 		DebugMode, | ||||||
|  | 		true, | ||||||
|  | 		func(router *Engine) { | ||||||
|  | 			router.LoadHTMLGlob("./testdata/template/*") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | 	defer ts.Close() | ||||||
|  | 
 | ||||||
| 	// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error | 	// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error | ||||||
| 	tr := &http.Transport{ | 	tr := &http.Transport{ | ||||||
| 		TLSClientConfig: &tls.Config{ | 		TLSClientConfig: &tls.Config{ | ||||||
| @ -128,29 +136,33 @@ func TestLoadHTMLGlobUsingTLS(t *testing.T) { | |||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 	client := &http.Client{Transport: tr} | 	client := &http.Client{Transport: tr} | ||||||
| 	res, err := client.Get("https://127.0.0.1:9999/test") | 	res, err := client.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLGlobFromFuncMap(t *testing.T) { | func TestLoadHTMLGlobFromFuncMap(t *testing.T) { | ||||||
| 	time.Now() | 	ts := setupHTMLFiles( | ||||||
| 	td := setupHTMLGlob(t, DebugMode, false) | 		t, | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/raw") | 		DebugMode, | ||||||
|  | 		false, | ||||||
|  | 		func(router *Engine) { | ||||||
|  | 			router.LoadHTMLGlob("./testdata/template/*") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | 	defer ts.Close() | ||||||
|  | 
 | ||||||
|  | 	res, err := http.Get(fmt.Sprintf("%s/raw", ts.URL)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "Date: 2017/07/01\n", string(resp[:])) | 	assert.Equal(t, "Date: 2017/07/01\n", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func init() { | func init() { | ||||||
| @ -164,59 +176,77 @@ func TestCreateEngine(t *testing.T) { | |||||||
| 	assert.Empty(t, router.Handlers) | 	assert.Empty(t, router.Handlers) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // func TestLoadHTMLDebugMode(t *testing.T) { | func TestLoadHTMLFilesTestMode(t *testing.T) { | ||||||
| // 	router := New() | 	ts := setupHTMLFiles( | ||||||
| // 	SetMode(DebugMode) | 		t, | ||||||
| // 	router.LoadHTMLGlob("*.testtmpl") | 		TestMode, | ||||||
| // 	r := router.HTMLRender.(render.HTMLDebug) | 		false, | ||||||
| // 	assert.Empty(t, r.Files) | 		func(router *Engine) { | ||||||
| // 	assert.Equal(t, "*.testtmpl", r.Glob) | 			router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl") | ||||||
| // | 		}, | ||||||
| // 	router.LoadHTMLFiles("index.html.testtmpl", "login.html.testtmpl") | 	) | ||||||
| // 	r = router.HTMLRender.(render.HTMLDebug) | 	defer ts.Close() | ||||||
| // 	assert.Empty(t, r.Glob) |  | ||||||
| // 	assert.Equal(t, []string{"index.html", "login.html"}, r.Files) |  | ||||||
| // 	SetMode(TestMode) |  | ||||||
| // } |  | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLFiles(t *testing.T) { | 	res, err := http.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	td := setupHTMLFiles(t, TestMode, false) |  | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") |  | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLFiles2(t *testing.T) { | func TestLoadHTMLFilesDebugMode(t *testing.T) { | ||||||
| 	td := setupHTMLFiles(t, DebugMode, false) | 	ts := setupHTMLFiles( | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") | 		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)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLFiles3(t *testing.T) { | func TestLoadHTMLFilesReleaseMode(t *testing.T) { | ||||||
| 	td := setupHTMLFiles(t, ReleaseMode, false) | 	ts := setupHTMLFiles( | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/test") | 		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)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLFilesUsingTLS(t *testing.T) { | func TestLoadHTMLFilesUsingTLS(t *testing.T) { | ||||||
| 	td := setupHTMLFiles(t, TestMode, true) | 	ts := setupHTMLFiles( | ||||||
|  | 		t, | ||||||
|  | 		TestMode, | ||||||
|  | 		true, | ||||||
|  | 		func(router *Engine) { | ||||||
|  | 			router.LoadHTMLFiles("./testdata/template/hello.tmpl", "./testdata/template/raw.tmpl") | ||||||
|  | 		}, | ||||||
|  | 	) | ||||||
|  | 	defer ts.Close() | ||||||
|  | 
 | ||||||
| 	// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error | 	// Use InsecureSkipVerify for avoiding `x509: certificate signed by unknown authority` error | ||||||
| 	tr := &http.Transport{ | 	tr := &http.Transport{ | ||||||
| 		TLSClientConfig: &tls.Config{ | 		TLSClientConfig: &tls.Config{ | ||||||
| @ -224,28 +254,33 @@ func TestLoadHTMLFilesUsingTLS(t *testing.T) { | |||||||
| 		}, | 		}, | ||||||
| 	} | 	} | ||||||
| 	client := &http.Client{Transport: tr} | 	client := &http.Client{Transport: tr} | ||||||
| 	res, err := client.Get("https://127.0.0.1:9999/test") | 	res, err := client.Get(fmt.Sprintf("%s/test", ts.URL)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "<h1>Hello world</h1>", string(resp[:])) | 	assert.Equal(t, "<h1>Hello world</h1>", string(resp)) | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestLoadHTMLFilesFuncMap(t *testing.T) { | func TestLoadHTMLFilesFuncMap(t *testing.T) { | ||||||
| 	time.Now() | 	ts := setupHTMLFiles( | ||||||
| 	td := setupHTMLFiles(t, TestMode, false) | 		t, | ||||||
| 	res, err := http.Get("http://127.0.0.1:8888/raw") | 		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)) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		fmt.Println(err) | 		fmt.Println(err) | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	resp, _ := ioutil.ReadAll(res.Body) | 	resp, _ := ioutil.ReadAll(res.Body) | ||||||
| 	assert.Equal(t, "Date: 2017/07/01\n", string(resp[:])) | 	assert.Equal(t, "Date: 2017/07/01\n", string(resp)) | ||||||
| 
 |  | ||||||
| 	td() |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestAddRoute(t *testing.T) { | func TestAddRoute(t *testing.T) { | ||||||
| @ -443,6 +478,60 @@ func TestListOfRoutes(t *testing.T) { | |||||||
| 	}) | 	}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 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) | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 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) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func assertRoutePresent(t *testing.T, gotRoutes RoutesInfo, wantRoute RouteInfo) { | func assertRoutePresent(t *testing.T, gotRoutes RoutesInfo, wantRoute RouteInfo) { | ||||||
| 	for _, gotRoute := range gotRoutes { | 	for _, gotRoute := range gotRoutes { | ||||||
| 		if gotRoute.Path == wantRoute.Path && gotRoute.Method == wantRoute.Method { | 		if gotRoute.Path == wantRoute.Path && gotRoute.Method == wantRoute.Method { | ||||||
|  | |||||||
| @ -285,6 +285,67 @@ var githubAPI = []route{ | |||||||
| 	{"DELETE", "/user/keys/:id"}, | 	{"DELETE", "/user/keys/:id"}, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestShouldBindUri(t *testing.T) { | ||||||
|  | 	DefaultWriter = os.Stdout | ||||||
|  | 	router := New() | ||||||
|  | 
 | ||||||
|  | 	type Person struct { | ||||||
|  | 		Name string `uri:"name" binding:"required"` | ||||||
|  | 		Id   string `uri:"id" binding:"required"` | ||||||
|  | 	} | ||||||
|  | 	router.Handle("GET", "/rest/:name/:id", func(c *Context) { | ||||||
|  | 		var person Person | ||||||
|  | 		assert.NoError(t, c.ShouldBindUri(&person)) | ||||||
|  | 		assert.True(t, "" != person.Name) | ||||||
|  | 		assert.True(t, "" != person.Id) | ||||||
|  | 		c.String(http.StatusOK, "ShouldBindUri test OK") | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	path, _ := exampleFromPath("/rest/:name/:id") | ||||||
|  | 	w := performRequest(router, "GET", path) | ||||||
|  | 	assert.Equal(t, "ShouldBindUri test OK", w.Body.String()) | ||||||
|  | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindUri(t *testing.T) { | ||||||
|  | 	DefaultWriter = os.Stdout | ||||||
|  | 	router := New() | ||||||
|  | 
 | ||||||
|  | 	type Person struct { | ||||||
|  | 		Name string `uri:"name" binding:"required"` | ||||||
|  | 		Id   string `uri:"id" binding:"required"` | ||||||
|  | 	} | ||||||
|  | 	router.Handle("GET", "/rest/:name/:id", func(c *Context) { | ||||||
|  | 		var person Person | ||||||
|  | 		assert.NoError(t, c.BindUri(&person)) | ||||||
|  | 		assert.True(t, "" != person.Name) | ||||||
|  | 		assert.True(t, "" != person.Id) | ||||||
|  | 		c.String(http.StatusOK, "BindUri test OK") | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	path, _ := exampleFromPath("/rest/:name/:id") | ||||||
|  | 	w := performRequest(router, "GET", path) | ||||||
|  | 	assert.Equal(t, "BindUri test OK", w.Body.String()) | ||||||
|  | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestBindUriError(t *testing.T) { | ||||||
|  | 	DefaultWriter = os.Stdout | ||||||
|  | 	router := New() | ||||||
|  | 
 | ||||||
|  | 	type Member struct { | ||||||
|  | 		Number string `uri:"num" binding:"required,uuid"` | ||||||
|  | 	} | ||||||
|  | 	router.Handle("GET", "/new/rest/:num", func(c *Context) { | ||||||
|  | 		var m Member | ||||||
|  | 		assert.Error(t, c.BindUri(&m)) | ||||||
|  | 	}) | ||||||
|  | 
 | ||||||
|  | 	path1, _ := exampleFromPath("/new/rest/:num") | ||||||
|  | 	w1 := performRequest(router, "GET", path1) | ||||||
|  | 	assert.Equal(t, http.StatusBadRequest, w1.Code) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func githubConfigRouter(router *Engine) { | func githubConfigRouter(router *Engine) { | ||||||
| 	for _, route := range githubAPI { | 	for _, route := range githubAPI { | ||||||
| 		router.Handle(route.method, route.path, func(c *Context) { | 		router.Handle(route.method, route.path, func(c *Context) { | ||||||
| @ -293,14 +354,14 @@ func githubConfigRouter(router *Engine) { | |||||||
| 			for _, param := range c.Params { | 			for _, param := range c.Params { | ||||||
| 				output[param.Key] = param.Value | 				output[param.Key] = param.Value | ||||||
| 			} | 			} | ||||||
| 			c.JSON(200, output) | 			c.JSON(http.StatusOK, output) | ||||||
| 		}) | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestGithubAPI(t *testing.T) { | func TestGithubAPI(t *testing.T) { | ||||||
| 	DefaultWriter = os.Stdout | 	DefaultWriter = os.Stdout | ||||||
| 	router := Default() | 	router := New() | ||||||
| 	githubConfigRouter(router) | 	githubConfigRouter(router) | ||||||
| 
 | 
 | ||||||
| 	for _, route := range githubAPI { | 	for _, route := range githubAPI { | ||||||
| @ -375,7 +436,7 @@ func BenchmarkParallelGithub(b *testing.B) { | |||||||
| 
 | 
 | ||||||
| func BenchmarkParallelGithubDefault(b *testing.B) { | func BenchmarkParallelGithubDefault(b *testing.B) { | ||||||
| 	DefaultWriter = os.Stdout | 	DefaultWriter = os.Stdout | ||||||
| 	router := Default() | 	router := New() | ||||||
| 	githubConfigRouter(router) | 	githubConfigRouter(router) | ||||||
| 
 | 
 | ||||||
| 	req, _ := http.NewRequest("POST", "/repos/manucorporat/sse/git/blobs", nil) | 	req, _ := http.NewRequest("POST", "/repos/manucorporat/sse/git/blobs", nil) | ||||||
|  | |||||||
							
								
								
									
										32
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								go.mod
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | |||||||
|  | module github.com/gin-gonic/gin | ||||||
|  | 
 | ||||||
|  | require ( | ||||||
|  | 	github.com/gin-contrib/sse v0.0.0-20190124093953-61b50c2ef482 | ||||||
|  | 	github.com/golang/protobuf v1.2.0 | ||||||
|  | 	github.com/json-iterator/go v1.1.5 | ||||||
|  | 	github.com/mattn/go-isatty v0.0.4 | ||||||
|  | 	github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect | ||||||
|  | 	github.com/modern-go/reflect2 v1.0.1 // indirect | ||||||
|  | 	github.com/stretchr/testify v1.3.0 | ||||||
|  | 	github.com/ugorji/go/codec v0.0.0-20181209151446-772ced7fd4c2 | ||||||
|  | 	golang.org/x/net v0.0.0-20190119204137-ed066c81e75e | ||||||
|  | 	golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 | ||||||
|  | 	golang.org/x/sys v0.0.0-20190124100055-b90733256f2e // indirect | ||||||
|  | 	gopkg.in/go-playground/assert.v1 v1.2.1 // indirect | ||||||
|  | 	gopkg.in/go-playground/validator.v8 v8.18.2 | ||||||
|  | 	gopkg.in/yaml.v2 v2.2.2 | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | exclude ( | ||||||
|  | 	github.com/campoy/embedmd v0.0.0-20181127031020-97c13d6e4160 | ||||||
|  | 	github.com/client9/misspell v0.3.4 | ||||||
|  | 	github.com/dustin/go-broadcast v0.0.0-20171205050544-f664265f5a66 | ||||||
|  | 	github.com/gin-gonic/autotls v0.0.0-20190119125636-0b5f4fc15768 | ||||||
|  | 	github.com/jessevdk/go-assets v0.0.0-20160921144138-4f4301a06e15 | ||||||
|  | 	github.com/manucorporat/stats v0.0.0-20180402194714-3ba42d56d227 | ||||||
|  | 	github.com/newrelic/go-agent v2.5.0+incompatible | ||||||
|  | 	github.com/thinkerou/favicon v0.1.0 | ||||||
|  | 	golang.org/x/crypto v0.0.0-20190123085648-057139ce5d2b | ||||||
|  | 	golang.org/x/lint v0.0.0-20181217174547-8f45f776aaf1 | ||||||
|  | 	google.golang.org/grpc v1.18.0 | ||||||
|  | ) | ||||||
							
								
								
									
										54
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										54
									
								
								go.sum
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,54 @@ | |||||||
|  | cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= | ||||||
|  | github.com/campoy/embedmd v0.0.0-20181127031020-97c13d6e4160 h1:HJpuhXOHC4EkXDARsLjmXAV9FhlY6qFDnKI/MJM6eoE= | ||||||
|  | github.com/campoy/embedmd v0.0.0-20181127031020-97c13d6e4160/go.mod h1:oxyr9RCiSXg0M3VJ3ks0UGfp98BpSSGr0kpiX3MzVl8= | ||||||
|  | github.com/client9/misspell v0.3.4 h1:ta993UF76GwbvJcIo3Y68y/M3WxlpEHPWIGDkJYwzJI= | ||||||
|  | github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= | ||||||
|  | github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8= | ||||||
|  | github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= | ||||||
|  | github.com/gin-contrib/sse v0.0.0-20190124093953-61b50c2ef482 h1:iOz5sIQUvuOlpiC7Q6+MmJQpWnlneYX98QIGf+2m50Y= | ||||||
|  | github.com/gin-contrib/sse v0.0.0-20190124093953-61b50c2ef482/go.mod h1:VJ0WA2NBN22VlZ2dKZQPAPnyWw5XTlK1KymzLKsr59s= | ||||||
|  | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b h1:VKtxabqXZkF25pY9ekfRL6a582T4P37/31XEstQ5p58= | ||||||
|  | github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= | ||||||
|  | github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= | ||||||
|  | github.com/golang/protobuf v1.2.0 h1:P3YflyNX/ehuJFLhxviNdFxQPkGK5cDcApsge1SqnvM= | ||||||
|  | github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= | ||||||
|  | github.com/json-iterator/go v1.1.5/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= | ||||||
|  | github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= | ||||||
|  | github.com/mattn/go-isatty v0.0.4 h1:bnP0vzxcAdeI1zdubAl5PjU6zsERjGZb7raWodagDYs= | ||||||
|  | github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= | ||||||
|  | github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= | ||||||
|  | github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= | ||||||
|  | github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= | ||||||
|  | github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= | ||||||
|  | github.com/stretchr/testify v1.3.0 h1:TivCn/peBQ7UY8ooIcPgZFpTNSz0Q2U6UrFlUfqbe0Q= | ||||||
|  | github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= | ||||||
|  | github.com/ugorji/go/codec v0.0.0-20181209151446-772ced7fd4c2 h1:EICbibRW4JNKMcY+LsWmuwob+CRS1BmdRdjphAm9mH4= | ||||||
|  | github.com/ugorji/go/codec v0.0.0-20181209151446-772ced7fd4c2/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= | ||||||
|  | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3 h1:x/bBzNauLQAlE3fLku/xy92Y8QwKX5HZymrMz2IiKFc= | ||||||
|  | golang.org/x/lint v0.0.0-20181026193005-c67002cb31c3/go.mod h1:UVdnD1Gm6xHRNCYTkRU2/jEulfH38KcIWyp/GAMgvoE= | ||||||
|  | golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||||
|  | golang.org/x/net v0.0.0-20190119204137-ed066c81e75e h1:MDa3fSUp6MdYHouVmCCNz/zaH2a6CRcxY3VhT/K3C5Q= | ||||||
|  | golang.org/x/net v0.0.0-20190119204137-ed066c81e75e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= | ||||||
|  | golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= | ||||||
|  | golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
|  | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4 h1:YUO/7uOKsKeq9UokNS62b8FYywz3ker1l1vDZRCRefw= | ||||||
|  | golang.org/x/sync v0.0.0-20181221193216-37e7f081c4d4/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= | ||||||
|  | golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
|  | golang.org/x/sys v0.0.0-20190124100055-b90733256f2e h1:3GIlrlVLfkoipSReOMNAgApI0ajnalyLa/EZHHca/XI= | ||||||
|  | golang.org/x/sys v0.0.0-20190124100055-b90733256f2e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= | ||||||
|  | golang.org/x/text v0.3.0 h1:g61tztE5qeGQ89tm6NTjjM9VPIm088od1l6aSorWRWg= | ||||||
|  | golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= | ||||||
|  | golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= | ||||||
|  | google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= | ||||||
|  | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8 h1:Nw54tB0rB7hY/N0NQvRW8DG4Yk3Q6T9cu9RcFQDu1tc= | ||||||
|  | google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= | ||||||
|  | google.golang.org/grpc v1.18.0 h1:IZl7mfBGfbhYx2p2rKRtYgDFw6SBz+kclmxYrCksPPA= | ||||||
|  | google.golang.org/grpc v1.18.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= | ||||||
|  | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= | ||||||
|  | gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= | ||||||
|  | gopkg.in/go-playground/assert.v1 v1.2.1/go.mod h1:9RXL0bg/zibRAgZUYszZSwO/z8Y/a8bDuhia5mkpMnE= | ||||||
|  | gopkg.in/go-playground/validator.v8 v8.18.2 h1:lFB4DoMU6B626w8ny76MV7VX6W2VHct2GVOI3xgiMrQ= | ||||||
|  | gopkg.in/go-playground/validator.v8 v8.18.2/go.mod h1:RX2a/7Ha8BgOhfk7j780h4/u/RRjR0eouCJSH80/M2Y= | ||||||
|  | gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw= | ||||||
|  | gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= | ||||||
|  | honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= | ||||||
							
								
								
									
										20
									
								
								internal/json/json.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								internal/json/json.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,20 @@ | |||||||
|  | // Copyright 2017 Bo-Yi Wu.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build !jsoniter | ||||||
|  | 
 | ||||||
|  | package json | ||||||
|  | 
 | ||||||
|  | import "encoding/json" | ||||||
|  | 
 | ||||||
|  | var ( | ||||||
|  | 	// Marshal is exported by gin/json package. | ||||||
|  | 	Marshal = json.Marshal | ||||||
|  | 	// MarshalIndent is exported by gin/json package. | ||||||
|  | 	MarshalIndent = json.MarshalIndent | ||||||
|  | 	// NewDecoder is exported by gin/json package. | ||||||
|  | 	NewDecoder = json.NewDecoder | ||||||
|  | 	// NewEncoder is exported by gin/json package. | ||||||
|  | 	NewEncoder = json.NewEncoder | ||||||
|  | ) | ||||||
							
								
								
									
										21
									
								
								internal/json/jsoniter.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								internal/json/jsoniter.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | // Copyright 2017 Bo-Yi Wu.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build jsoniter | ||||||
|  | 
 | ||||||
|  | package json | ||||||
|  | 
 | ||||||
|  | import "github.com/json-iterator/go" | ||||||
|  | 
 | ||||||
|  | var ( | ||||||
|  | 	json = jsoniter.ConfigCompatibleWithStandardLibrary | ||||||
|  | 	// Marshal is exported by gin/json package. | ||||||
|  | 	Marshal = json.Marshal | ||||||
|  | 	// MarshalIndent is exported by gin/json package. | ||||||
|  | 	MarshalIndent = json.MarshalIndent | ||||||
|  | 	// NewDecoder is exported by gin/json package. | ||||||
|  | 	NewDecoder = json.NewDecoder | ||||||
|  | 	// NewEncoder is exported by gin/json package. | ||||||
|  | 	NewEncoder = json.NewEncoder | ||||||
|  | ) | ||||||
							
								
								
									
										15
									
								
								json/json.go
									
									
									
									
									
								
							
							
						
						
									
										15
									
								
								json/json.go
									
									
									
									
									
								
							| @ -1,15 +0,0 @@ | |||||||
| // Copyright 2017 Bo-Yi Wu.  All rights reserved. |  | ||||||
| // Use of this source code is governed by a MIT style |  | ||||||
| // license that can be found in the LICENSE file. |  | ||||||
| 
 |  | ||||||
| // +build !jsoniter |  | ||||||
| 
 |  | ||||||
| package json |  | ||||||
| 
 |  | ||||||
| import "encoding/json" |  | ||||||
| 
 |  | ||||||
| var ( |  | ||||||
| 	Marshal       = json.Marshal |  | ||||||
| 	MarshalIndent = json.MarshalIndent |  | ||||||
| 	NewDecoder    = json.NewDecoder |  | ||||||
| ) |  | ||||||
| @ -1,16 +0,0 @@ | |||||||
| // Copyright 2017 Bo-Yi Wu.  All rights reserved. |  | ||||||
| // Use of this source code is governed by a MIT style |  | ||||||
| // license that can be found in the LICENSE file. |  | ||||||
| 
 |  | ||||||
| // +build jsoniter |  | ||||||
| 
 |  | ||||||
| package json |  | ||||||
| 
 |  | ||||||
| import "github.com/json-iterator/go" |  | ||||||
| 
 |  | ||||||
| var ( |  | ||||||
| 	json          = jsoniter.ConfigCompatibleWithStandardLibrary |  | ||||||
| 	Marshal       = json.Marshal |  | ||||||
| 	MarshalIndent = json.MarshalIndent |  | ||||||
| 	NewDecoder    = json.NewDecoder |  | ||||||
| ) |  | ||||||
							
								
								
									
										146
									
								
								logger.go
									
									
									
									
									
								
							
							
						
						
									
										146
									
								
								logger.go
									
									
									
									
									
								
							| @ -7,6 +7,7 @@ package gin | |||||||
| import ( | import ( | ||||||
| 	"fmt" | 	"fmt" | ||||||
| 	"io" | 	"io" | ||||||
|  | 	"net/http" | ||||||
| 	"os" | 	"os" | ||||||
| 	"time" | 	"time" | ||||||
| 
 | 
 | ||||||
| @ -16,20 +17,85 @@ import ( | |||||||
| var ( | var ( | ||||||
| 	green        = string([]byte{27, 91, 57, 55, 59, 52, 50, 109}) | 	green        = string([]byte{27, 91, 57, 55, 59, 52, 50, 109}) | ||||||
| 	white        = string([]byte{27, 91, 57, 48, 59, 52, 55, 109}) | 	white        = string([]byte{27, 91, 57, 48, 59, 52, 55, 109}) | ||||||
| 	yellow       = string([]byte{27, 91, 57, 55, 59, 52, 51, 109}) | 	yellow       = string([]byte{27, 91, 57, 48, 59, 52, 51, 109}) | ||||||
| 	red          = string([]byte{27, 91, 57, 55, 59, 52, 49, 109}) | 	red          = string([]byte{27, 91, 57, 55, 59, 52, 49, 109}) | ||||||
| 	blue         = string([]byte{27, 91, 57, 55, 59, 52, 52, 109}) | 	blue         = string([]byte{27, 91, 57, 55, 59, 52, 52, 109}) | ||||||
| 	magenta      = string([]byte{27, 91, 57, 55, 59, 52, 53, 109}) | 	magenta      = string([]byte{27, 91, 57, 55, 59, 52, 53, 109}) | ||||||
| 	cyan         = string([]byte{27, 91, 57, 55, 59, 52, 54, 109}) | 	cyan         = string([]byte{27, 91, 57, 55, 59, 52, 54, 109}) | ||||||
| 	reset        = string([]byte{27, 91, 48, 109}) | 	reset        = string([]byte{27, 91, 48, 109}) | ||||||
| 	disableColor = false | 	disableColor = false | ||||||
|  | 	forceColor   = false | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // LoggerConfig defines the config for Logger middleware. | ||||||
|  | type LoggerConfig struct { | ||||||
|  | 	// Optional. Default value is gin.defaultLogFormatter | ||||||
|  | 	Formatter LogFormatter | ||||||
|  | 
 | ||||||
|  | 	// Output is a writer where logs are written. | ||||||
|  | 	// Optional. Default value is gin.DefaultWriter. | ||||||
|  | 	Output io.Writer | ||||||
|  | 
 | ||||||
|  | 	// SkipPaths is a url path array which logs are not written. | ||||||
|  | 	// Optional. | ||||||
|  | 	SkipPaths []string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // LogFormatter gives the signature of the formatter function passed to LoggerWithFormatter | ||||||
|  | type LogFormatter func(params LogFormatterParams) string | ||||||
|  | 
 | ||||||
|  | // LogFormatterParams is the structure any formatter will be handed when time to log comes | ||||||
|  | type LogFormatterParams struct { | ||||||
|  | 	Request *http.Request | ||||||
|  | 
 | ||||||
|  | 	// TimeStamp shows the time after the server returns a response. | ||||||
|  | 	TimeStamp time.Time | ||||||
|  | 	// StatusCode is HTTP response code. | ||||||
|  | 	StatusCode int | ||||||
|  | 	// Latency is how much time the server cost to process a certain request. | ||||||
|  | 	Latency time.Duration | ||||||
|  | 	// ClientIP equals Context's ClientIP method. | ||||||
|  | 	ClientIP string | ||||||
|  | 	// Method is the HTTP method given to the request. | ||||||
|  | 	Method string | ||||||
|  | 	// Path is a path the client requests. | ||||||
|  | 	Path string | ||||||
|  | 	// ErrorMessage is set if error has occurred in processing the request. | ||||||
|  | 	ErrorMessage string | ||||||
|  | 	// IsTerm shows whether does gin's output descriptor refers to a terminal. | ||||||
|  | 	IsTerm bool | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // defaultLogFormatter is the default log format function Logger middleware uses. | ||||||
|  | var defaultLogFormatter = func(param LogFormatterParams) string { | ||||||
|  | 	var statusColor, methodColor, resetColor string | ||||||
|  | 	if param.IsTerm { | ||||||
|  | 		statusColor = colorForStatus(param.StatusCode) | ||||||
|  | 		methodColor = colorForMethod(param.Method) | ||||||
|  | 		resetColor = reset | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return fmt.Sprintf("[GIN] %v |%s %3d %s| %13v | %15s |%s %-7s %s %s\n%s", | ||||||
|  | 		param.TimeStamp.Format("2006/01/02 - 15:04:05"), | ||||||
|  | 		statusColor, param.StatusCode, resetColor, | ||||||
|  | 		param.Latency, | ||||||
|  | 		param.ClientIP, | ||||||
|  | 		methodColor, param.Method, resetColor, | ||||||
|  | 		param.Path, | ||||||
|  | 		param.ErrorMessage, | ||||||
|  | 	) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // DisableConsoleColor disables color output in the console. | // DisableConsoleColor disables color output in the console. | ||||||
| func DisableConsoleColor() { | func DisableConsoleColor() { | ||||||
| 	disableColor = true | 	disableColor = true | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // ForceConsoleColor force color output in the console. | ||||||
|  | func ForceConsoleColor() { | ||||||
|  | 	forceColor = true | ||||||
|  | } | ||||||
|  | 
 | ||||||
| // ErrorLogger returns a handlerfunc for any error type. | // ErrorLogger returns a handlerfunc for any error type. | ||||||
| func ErrorLogger() HandlerFunc { | func ErrorLogger() HandlerFunc { | ||||||
| 	return ErrorLoggerT(ErrorTypeAny) | 	return ErrorLoggerT(ErrorTypeAny) | ||||||
| @ -49,17 +115,44 @@ func ErrorLoggerT(typ ErrorType) HandlerFunc { | |||||||
| // Logger instances a Logger middleware that will write the logs to gin.DefaultWriter. | // Logger instances a Logger middleware that will write the logs to gin.DefaultWriter. | ||||||
| // By default gin.DefaultWriter = os.Stdout. | // By default gin.DefaultWriter = os.Stdout. | ||||||
| func Logger() HandlerFunc { | func Logger() HandlerFunc { | ||||||
| 	return LoggerWithWriter(DefaultWriter) | 	return LoggerWithConfig(LoggerConfig{}) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // LoggerWithWriter instance a Logger middleware with the specified writter buffer. | // LoggerWithFormatter instance a Logger middleware with the specified log format function. | ||||||
|  | func LoggerWithFormatter(f LogFormatter) HandlerFunc { | ||||||
|  | 	return LoggerWithConfig(LoggerConfig{ | ||||||
|  | 		Formatter: f, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // LoggerWithWriter instance a Logger middleware with the specified writer buffer. | ||||||
| // Example: os.Stdout, a file opened in write mode, a socket... | // Example: os.Stdout, a file opened in write mode, a socket... | ||||||
| func LoggerWithWriter(out io.Writer, notlogged ...string) HandlerFunc { | func LoggerWithWriter(out io.Writer, notlogged ...string) HandlerFunc { | ||||||
|  | 	return LoggerWithConfig(LoggerConfig{ | ||||||
|  | 		Output:    out, | ||||||
|  | 		SkipPaths: notlogged, | ||||||
|  | 	}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // LoggerWithConfig instance a Logger middleware with config. | ||||||
|  | func LoggerWithConfig(conf LoggerConfig) HandlerFunc { | ||||||
|  | 	formatter := conf.Formatter | ||||||
|  | 	if formatter == nil { | ||||||
|  | 		formatter = defaultLogFormatter | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	out := conf.Output | ||||||
|  | 	if out == nil { | ||||||
|  | 		out = DefaultWriter | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	notlogged := conf.SkipPaths | ||||||
|  | 
 | ||||||
| 	isTerm := true | 	isTerm := true | ||||||
| 
 | 
 | ||||||
| 	if w, ok := out.(*os.File); !ok || | 	if w, ok := out.(*os.File); (!ok || | ||||||
| 		(os.Getenv("TERM") == "dumb" || (!isatty.IsTerminal(w.Fd()) && !isatty.IsCygwinTerminal(w.Fd()))) || | 		(os.Getenv("TERM") == "dumb" || (!isatty.IsTerminal(w.Fd()) && !isatty.IsCygwinTerminal(w.Fd()))) || | ||||||
| 		disableColor { | 		disableColor) && !forceColor { | ||||||
| 		isTerm = false | 		isTerm = false | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -84,45 +177,38 @@ func LoggerWithWriter(out io.Writer, notlogged ...string) HandlerFunc { | |||||||
| 
 | 
 | ||||||
| 		// Log only when path is not being skipped | 		// Log only when path is not being skipped | ||||||
| 		if _, ok := skip[path]; !ok { | 		if _, ok := skip[path]; !ok { | ||||||
| 			// Stop timer | 			param := LogFormatterParams{ | ||||||
| 			end := time.Now() | 				Request: c.Request, | ||||||
| 			latency := end.Sub(start) | 				IsTerm:  isTerm, | ||||||
| 
 |  | ||||||
| 			clientIP := c.ClientIP() |  | ||||||
| 			method := c.Request.Method |  | ||||||
| 			statusCode := c.Writer.Status() |  | ||||||
| 			var statusColor, methodColor, resetColor string |  | ||||||
| 			if isTerm { |  | ||||||
| 				statusColor = colorForStatus(statusCode) |  | ||||||
| 				methodColor = colorForMethod(method) |  | ||||||
| 				resetColor = reset |  | ||||||
| 			} | 			} | ||||||
| 			comment := c.Errors.ByType(ErrorTypePrivate).String() | 
 | ||||||
|  | 			// Stop timer | ||||||
|  | 			param.TimeStamp = time.Now() | ||||||
|  | 			param.Latency = param.TimeStamp.Sub(start) | ||||||
|  | 
 | ||||||
|  | 			param.ClientIP = c.ClientIP() | ||||||
|  | 			param.Method = c.Request.Method | ||||||
|  | 			param.StatusCode = c.Writer.Status() | ||||||
|  | 			param.ErrorMessage = c.Errors.ByType(ErrorTypePrivate).String() | ||||||
| 
 | 
 | ||||||
| 			if raw != "" { | 			if raw != "" { | ||||||
| 				path = path + "?" + raw | 				path = path + "?" + raw | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			fmt.Fprintf(out, "[GIN] %v |%s %3d %s| %13v | %15s |%s %-7s %s %s\n%s", | 			param.Path = path | ||||||
| 				end.Format("2006/01/02 - 15:04:05"), | 
 | ||||||
| 				statusColor, statusCode, resetColor, | 			fmt.Fprint(out, formatter(param)) | ||||||
| 				latency, |  | ||||||
| 				clientIP, |  | ||||||
| 				methodColor, method, resetColor, |  | ||||||
| 				path, |  | ||||||
| 				comment, |  | ||||||
| 			) |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func colorForStatus(code int) string { | func colorForStatus(code int) string { | ||||||
| 	switch { | 	switch { | ||||||
| 	case code >= 200 && code < 300: | 	case code >= http.StatusOK && code < http.StatusMultipleChoices: | ||||||
| 		return green | 		return green | ||||||
| 	case code >= 300 && code < 400: | 	case code >= http.StatusMultipleChoices && code < http.StatusBadRequest: | ||||||
| 		return white | 		return white | ||||||
| 	case code >= 400 && code < 500: | 	case code >= http.StatusBadRequest && code < http.StatusInternalServerError: | ||||||
| 		return yellow | 		return yellow | ||||||
| 	default: | 	default: | ||||||
| 		return red | 		return red | ||||||
|  | |||||||
							
								
								
									
										224
									
								
								logger_test.go
									
									
									
									
									
								
							
							
						
						
									
										224
									
								
								logger_test.go
									
									
									
									
									
								
							| @ -7,7 +7,10 @@ package gin | |||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"net/http" | ||||||
| 	"testing" | 	"testing" | ||||||
|  | 	"time" | ||||||
| 
 | 
 | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| ) | ) | ||||||
| @ -78,13 +81,185 @@ func TestLogger(t *testing.T) { | |||||||
| 	assert.Contains(t, buffer.String(), "404") | 	assert.Contains(t, buffer.String(), "404") | ||||||
| 	assert.Contains(t, buffer.String(), "GET") | 	assert.Contains(t, buffer.String(), "GET") | ||||||
| 	assert.Contains(t, buffer.String(), "/notfound") | 	assert.Contains(t, buffer.String(), "/notfound") | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|  | func TestLoggerWithConfig(t *testing.T) { | ||||||
|  | 	buffer := new(bytes.Buffer) | ||||||
|  | 	router := New() | ||||||
|  | 	router.Use(LoggerWithConfig(LoggerConfig{Output: buffer})) | ||||||
|  | 	router.GET("/example", func(c *Context) {}) | ||||||
|  | 	router.POST("/example", func(c *Context) {}) | ||||||
|  | 	router.PUT("/example", func(c *Context) {}) | ||||||
|  | 	router.DELETE("/example", func(c *Context) {}) | ||||||
|  | 	router.PATCH("/example", func(c *Context) {}) | ||||||
|  | 	router.HEAD("/example", func(c *Context) {}) | ||||||
|  | 	router.OPTIONS("/example", func(c *Context) {}) | ||||||
|  | 
 | ||||||
|  | 	performRequest(router, "GET", "/example?a=100") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "GET") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "a=100") | ||||||
|  | 
 | ||||||
|  | 	// I wrote these first (extending the above) but then realized they are more | ||||||
|  | 	// like integration tests because they test the whole logging process rather | ||||||
|  | 	// than individual functions.  Im not sure where these should go. | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "POST", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "POST") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "PUT", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "PUT") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "DELETE", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "DELETE") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "PATCH", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "PATCH") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "HEAD", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "HEAD") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "OPTIONS", "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "OPTIONS") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "GET", "/notfound") | ||||||
|  | 	assert.Contains(t, buffer.String(), "404") | ||||||
|  | 	assert.Contains(t, buffer.String(), "GET") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/notfound") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestLoggerWithFormatter(t *testing.T) { | ||||||
|  | 	buffer := new(bytes.Buffer) | ||||||
|  | 
 | ||||||
|  | 	d := DefaultWriter | ||||||
|  | 	DefaultWriter = buffer | ||||||
|  | 	defer func() { | ||||||
|  | 		DefaultWriter = d | ||||||
|  | 	}() | ||||||
|  | 
 | ||||||
|  | 	router := New() | ||||||
|  | 	router.Use(LoggerWithFormatter(func(param LogFormatterParams) string { | ||||||
|  | 		return fmt.Sprintf("[FORMATTER TEST] %v | %3d | %13v | %15s | %-7s %s\n%s", | ||||||
|  | 			param.TimeStamp.Format("2006/01/02 - 15:04:05"), | ||||||
|  | 			param.StatusCode, | ||||||
|  | 			param.Latency, | ||||||
|  | 			param.ClientIP, | ||||||
|  | 			param.Method, | ||||||
|  | 			param.Path, | ||||||
|  | 			param.ErrorMessage, | ||||||
|  | 		) | ||||||
|  | 	})) | ||||||
|  | 	router.GET("/example", func(c *Context) {}) | ||||||
|  | 	performRequest(router, "GET", "/example?a=100") | ||||||
|  | 
 | ||||||
|  | 	// output test | ||||||
|  | 	assert.Contains(t, buffer.String(), "[FORMATTER TEST]") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "GET") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "a=100") | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestLoggerWithConfigFormatting(t *testing.T) { | ||||||
|  | 	var gotParam LogFormatterParams | ||||||
|  | 	buffer := new(bytes.Buffer) | ||||||
|  | 
 | ||||||
|  | 	router := New() | ||||||
|  | 	router.Use(LoggerWithConfig(LoggerConfig{ | ||||||
|  | 		Output: buffer, | ||||||
|  | 		Formatter: func(param LogFormatterParams) string { | ||||||
|  | 			// for assert test | ||||||
|  | 			gotParam = param | ||||||
|  | 
 | ||||||
|  | 			return fmt.Sprintf("[FORMATTER TEST] %v | %3d | %13v | %15s | %-7s %s\n%s", | ||||||
|  | 				param.TimeStamp.Format("2006/01/02 - 15:04:05"), | ||||||
|  | 				param.StatusCode, | ||||||
|  | 				param.Latency, | ||||||
|  | 				param.ClientIP, | ||||||
|  | 				param.Method, | ||||||
|  | 				param.Path, | ||||||
|  | 				param.ErrorMessage, | ||||||
|  | 			) | ||||||
|  | 		}, | ||||||
|  | 	})) | ||||||
|  | 	router.GET("/example", func(c *Context) { | ||||||
|  | 		// set dummy ClientIP | ||||||
|  | 		c.Request.Header.Set("X-Forwarded-For", "20.20.20.20") | ||||||
|  | 	}) | ||||||
|  | 	performRequest(router, "GET", "/example?a=100") | ||||||
|  | 
 | ||||||
|  | 	// output test | ||||||
|  | 	assert.Contains(t, buffer.String(), "[FORMATTER TEST]") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 	assert.Contains(t, buffer.String(), "GET") | ||||||
|  | 	assert.Contains(t, buffer.String(), "/example") | ||||||
|  | 	assert.Contains(t, buffer.String(), "a=100") | ||||||
|  | 
 | ||||||
|  | 	// LogFormatterParams test | ||||||
|  | 	assert.NotNil(t, gotParam.Request) | ||||||
|  | 	assert.NotEmpty(t, gotParam.TimeStamp) | ||||||
|  | 	assert.Equal(t, 200, gotParam.StatusCode) | ||||||
|  | 	assert.NotEmpty(t, gotParam.Latency) | ||||||
|  | 	assert.Equal(t, "20.20.20.20", gotParam.ClientIP) | ||||||
|  | 	assert.Equal(t, "GET", gotParam.Method) | ||||||
|  | 	assert.Equal(t, "/example?a=100", gotParam.Path) | ||||||
|  | 	assert.Empty(t, gotParam.ErrorMessage) | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestDefaultLogFormatter(t *testing.T) { | ||||||
|  | 	timeStamp := time.Unix(1544173902, 0).UTC() | ||||||
|  | 
 | ||||||
|  | 	termFalseParam := LogFormatterParams{ | ||||||
|  | 		TimeStamp:    timeStamp, | ||||||
|  | 		StatusCode:   200, | ||||||
|  | 		Latency:      time.Second * 5, | ||||||
|  | 		ClientIP:     "20.20.20.20", | ||||||
|  | 		Method:       "GET", | ||||||
|  | 		Path:         "/", | ||||||
|  | 		ErrorMessage: "", | ||||||
|  | 		IsTerm:       false, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	termTrueParam := LogFormatterParams{ | ||||||
|  | 		TimeStamp:    timeStamp, | ||||||
|  | 		StatusCode:   200, | ||||||
|  | 		Latency:      time.Second * 5, | ||||||
|  | 		ClientIP:     "20.20.20.20", | ||||||
|  | 		Method:       "GET", | ||||||
|  | 		Path:         "/", | ||||||
|  | 		ErrorMessage: "", | ||||||
|  | 		IsTerm:       true, | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	assert.Equal(t, "[GIN] 2018/12/07 - 09:11:42 | 200 |            5s |     20.20.20.20 | GET      /\n", defaultLogFormatter(termFalseParam)) | ||||||
|  | 
 | ||||||
|  | 	assert.Equal(t, "[GIN] 2018/12/07 - 09:11:42 |\x1b[97;42m 200 \x1b[0m|            5s |     20.20.20.20 |\x1b[97;44m GET     \x1b[0m /\n", defaultLogFormatter(termTrueParam)) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestColorForMethod(t *testing.T) { | func TestColorForMethod(t *testing.T) { | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 52, 109}), colorForMethod("GET"), "get should be blue") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 52, 109}), colorForMethod("GET"), "get should be blue") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 54, 109}), colorForMethod("POST"), "post should be cyan") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 54, 109}), colorForMethod("POST"), "post should be cyan") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 51, 109}), colorForMethod("PUT"), "put should be yellow") | 	assert.Equal(t, string([]byte{27, 91, 57, 48, 59, 52, 51, 109}), colorForMethod("PUT"), "put should be yellow") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 49, 109}), colorForMethod("DELETE"), "delete should be red") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 49, 109}), colorForMethod("DELETE"), "delete should be red") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 50, 109}), colorForMethod("PATCH"), "patch should be green") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 50, 109}), colorForMethod("PATCH"), "patch should be green") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 53, 109}), colorForMethod("HEAD"), "head should be magenta") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 53, 109}), colorForMethod("HEAD"), "head should be magenta") | ||||||
| @ -93,9 +268,9 @@ func TestColorForMethod(t *testing.T) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestColorForStatus(t *testing.T) { | func TestColorForStatus(t *testing.T) { | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 50, 109}), colorForStatus(200), "2xx should be green") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 50, 109}), colorForStatus(http.StatusOK), "2xx should be green") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 48, 59, 52, 55, 109}), colorForStatus(301), "3xx should be white") | 	assert.Equal(t, string([]byte{27, 91, 57, 48, 59, 52, 55, 109}), colorForStatus(http.StatusMovedPermanently), "3xx should be white") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 51, 109}), colorForStatus(404), "4xx should be yellow") | 	assert.Equal(t, string([]byte{27, 91, 57, 48, 59, 52, 51, 109}), colorForStatus(http.StatusNotFound), "4xx should be yellow") | ||||||
| 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 49, 109}), colorForStatus(2), "other things should be red") | 	assert.Equal(t, string([]byte{27, 91, 57, 55, 59, 52, 49, 109}), colorForStatus(2), "other things should be red") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -103,30 +278,30 @@ func TestErrorLogger(t *testing.T) { | |||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(ErrorLogger()) | 	router.Use(ErrorLogger()) | ||||||
| 	router.GET("/error", func(c *Context) { | 	router.GET("/error", func(c *Context) { | ||||||
| 		c.Error(errors.New("this is an error")) | 		c.Error(errors.New("this is an error")) // nolint: errcheck | ||||||
| 	}) | 	}) | ||||||
| 	router.GET("/abort", func(c *Context) { | 	router.GET("/abort", func(c *Context) { | ||||||
| 		c.AbortWithError(401, errors.New("no authorized")) | 		c.AbortWithError(http.StatusUnauthorized, errors.New("no authorized")) // nolint: errcheck | ||||||
| 	}) | 	}) | ||||||
| 	router.GET("/print", func(c *Context) { | 	router.GET("/print", func(c *Context) { | ||||||
| 		c.Error(errors.New("this is an error")) | 		c.Error(errors.New("this is an error")) // nolint: errcheck | ||||||
| 		c.String(500, "hola!") | 		c.String(http.StatusInternalServerError, "hola!") | ||||||
| 	}) | 	}) | ||||||
| 
 | 
 | ||||||
| 	w := performRequest(router, "GET", "/error") | 	w := performRequest(router, "GET", "/error") | ||||||
| 	assert.Equal(t, 200, w.Code) | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
| 	assert.Equal(t, "{\"error\":\"this is an error\"}", w.Body.String()) | 	assert.Equal(t, "{\"error\":\"this is an error\"}", w.Body.String()) | ||||||
| 
 | 
 | ||||||
| 	w = performRequest(router, "GET", "/abort") | 	w = performRequest(router, "GET", "/abort") | ||||||
| 	assert.Equal(t, 401, w.Code) | 	assert.Equal(t, http.StatusUnauthorized, w.Code) | ||||||
| 	assert.Equal(t, "{\"error\":\"no authorized\"}", w.Body.String()) | 	assert.Equal(t, "{\"error\":\"no authorized\"}", w.Body.String()) | ||||||
| 
 | 
 | ||||||
| 	w = performRequest(router, "GET", "/print") | 	w = performRequest(router, "GET", "/print") | ||||||
| 	assert.Equal(t, 500, w.Code) | 	assert.Equal(t, http.StatusInternalServerError, w.Code) | ||||||
| 	assert.Equal(t, "hola!{\"error\":\"this is an error\"}", w.Body.String()) | 	assert.Equal(t, "hola!{\"error\":\"this is an error\"}", w.Body.String()) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestSkippingPaths(t *testing.T) { | func TestLoggerWithWriterSkippingPaths(t *testing.T) { | ||||||
| 	buffer := new(bytes.Buffer) | 	buffer := new(bytes.Buffer) | ||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(LoggerWithWriter(buffer, "/skipped")) | 	router.Use(LoggerWithWriter(buffer, "/skipped")) | ||||||
| @ -141,9 +316,34 @@ func TestSkippingPaths(t *testing.T) { | |||||||
| 	assert.Contains(t, buffer.String(), "") | 	assert.Contains(t, buffer.String(), "") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | func TestLoggerWithConfigSkippingPaths(t *testing.T) { | ||||||
|  | 	buffer := new(bytes.Buffer) | ||||||
|  | 	router := New() | ||||||
|  | 	router.Use(LoggerWithConfig(LoggerConfig{ | ||||||
|  | 		Output:    buffer, | ||||||
|  | 		SkipPaths: []string{"/skipped"}, | ||||||
|  | 	})) | ||||||
|  | 	router.GET("/logged", func(c *Context) {}) | ||||||
|  | 	router.GET("/skipped", func(c *Context) {}) | ||||||
|  | 
 | ||||||
|  | 	performRequest(router, "GET", "/logged") | ||||||
|  | 	assert.Contains(t, buffer.String(), "200") | ||||||
|  | 
 | ||||||
|  | 	buffer.Reset() | ||||||
|  | 	performRequest(router, "GET", "/skipped") | ||||||
|  | 	assert.Contains(t, buffer.String(), "") | ||||||
|  | } | ||||||
|  | 
 | ||||||
| func TestDisableConsoleColor(t *testing.T) { | func TestDisableConsoleColor(t *testing.T) { | ||||||
| 	New() | 	New() | ||||||
| 	assert.False(t, disableColor) | 	assert.False(t, disableColor) | ||||||
| 	DisableConsoleColor() | 	DisableConsoleColor() | ||||||
| 	assert.True(t, disableColor) | 	assert.True(t, disableColor) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func TestForceConsoleColor(t *testing.T) { | ||||||
|  | 	New() | ||||||
|  | 	assert.False(t, forceColor) | ||||||
|  | 	ForceConsoleColor() | ||||||
|  | 	assert.True(t, forceColor) | ||||||
|  | } | ||||||
|  | |||||||
| @ -6,6 +6,7 @@ package gin | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"errors" | 	"errors" | ||||||
|  | 	"net/http" | ||||||
| 	"strings" | 	"strings" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
| @ -37,7 +38,7 @@ func TestMiddlewareGeneralCase(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 200, w.Code) | 	assert.Equal(t, http.StatusOK, w.Code) | ||||||
| 	assert.Equal(t, "ACDB", signature) | 	assert.Equal(t, "ACDB", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -73,7 +74,7 @@ func TestMiddlewareNoRoute(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 404, w.Code) | 	assert.Equal(t, http.StatusNotFound, w.Code) | ||||||
| 	assert.Equal(t, "ACEGHFDB", signature) | 	assert.Equal(t, "ACEGHFDB", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -110,7 +111,7 @@ func TestMiddlewareNoMethodEnabled(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 405, w.Code) | 	assert.Equal(t, http.StatusMethodNotAllowed, w.Code) | ||||||
| 	assert.Equal(t, "ACEGHFDB", signature) | 	assert.Equal(t, "ACEGHFDB", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -147,7 +148,7 @@ func TestMiddlewareNoMethodDisabled(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 404, w.Code) | 	assert.Equal(t, http.StatusNotFound, w.Code) | ||||||
| 	assert.Equal(t, "AC X DB", signature) | 	assert.Equal(t, "AC X DB", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -159,7 +160,7 @@ func TestMiddlewareAbort(t *testing.T) { | |||||||
| 	}) | 	}) | ||||||
| 	router.Use(func(c *Context) { | 	router.Use(func(c *Context) { | ||||||
| 		signature += "C" | 		signature += "C" | ||||||
| 		c.AbortWithStatus(401) | 		c.AbortWithStatus(http.StatusUnauthorized) | ||||||
| 		c.Next() | 		c.Next() | ||||||
| 		signature += "D" | 		signature += "D" | ||||||
| 	}) | 	}) | ||||||
| @ -173,7 +174,7 @@ func TestMiddlewareAbort(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 401, w.Code) | 	assert.Equal(t, http.StatusUnauthorized, w.Code) | ||||||
| 	assert.Equal(t, "ACD", signature) | 	assert.Equal(t, "ACD", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -183,7 +184,7 @@ func TestMiddlewareAbortHandlersChainAndNext(t *testing.T) { | |||||||
| 	router.Use(func(c *Context) { | 	router.Use(func(c *Context) { | ||||||
| 		signature += "A" | 		signature += "A" | ||||||
| 		c.Next() | 		c.Next() | ||||||
| 		c.AbortWithStatus(410) | 		c.AbortWithStatus(http.StatusGone) | ||||||
| 		signature += "B" | 		signature += "B" | ||||||
| 
 | 
 | ||||||
| 	}) | 	}) | ||||||
| @ -195,7 +196,7 @@ func TestMiddlewareAbortHandlersChainAndNext(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 410, w.Code) | 	assert.Equal(t, http.StatusGone, w.Code) | ||||||
| 	assert.Equal(t, "ACB", signature) | 	assert.Equal(t, "ACB", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -207,7 +208,7 @@ func TestMiddlewareFailHandlersChain(t *testing.T) { | |||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(func(context *Context) { | 	router.Use(func(context *Context) { | ||||||
| 		signature += "A" | 		signature += "A" | ||||||
| 		context.AbortWithError(500, errors.New("foo")) | 		context.AbortWithError(http.StatusInternalServerError, errors.New("foo")) // nolint: errcheck | ||||||
| 	}) | 	}) | ||||||
| 	router.Use(func(context *Context) { | 	router.Use(func(context *Context) { | ||||||
| 		signature += "B" | 		signature += "B" | ||||||
| @ -218,25 +219,25 @@ func TestMiddlewareFailHandlersChain(t *testing.T) { | |||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 500, w.Code) | 	assert.Equal(t, http.StatusInternalServerError, w.Code) | ||||||
| 	assert.Equal(t, "A", signature) | 	assert.Equal(t, "A", signature) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestMiddlewareWrite(t *testing.T) { | func TestMiddlewareWrite(t *testing.T) { | ||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(func(c *Context) { | 	router.Use(func(c *Context) { | ||||||
| 		c.String(400, "hola\n") | 		c.String(http.StatusBadRequest, "hola\n") | ||||||
| 	}) | 	}) | ||||||
| 	router.Use(func(c *Context) { | 	router.Use(func(c *Context) { | ||||||
| 		c.XML(400, H{"foo": "bar"}) | 		c.XML(http.StatusBadRequest, H{"foo": "bar"}) | ||||||
| 	}) | 	}) | ||||||
| 	router.Use(func(c *Context) { | 	router.Use(func(c *Context) { | ||||||
| 		c.JSON(400, H{"foo": "bar"}) | 		c.JSON(http.StatusBadRequest, H{"foo": "bar"}) | ||||||
| 	}) | 	}) | ||||||
| 	router.GET("/", func(c *Context) { | 	router.GET("/", func(c *Context) { | ||||||
| 		c.JSON(400, H{"foo": "bar"}) | 		c.JSON(http.StatusBadRequest, H{"foo": "bar"}) | ||||||
| 	}, func(c *Context) { | 	}, func(c *Context) { | ||||||
| 		c.Render(400, sse.Event{ | 		c.Render(http.StatusBadRequest, sse.Event{ | ||||||
| 			Event: "test", | 			Event: "test", | ||||||
| 			Data:  "message", | 			Data:  "message", | ||||||
| 		}) | 		}) | ||||||
| @ -244,6 +245,6 @@ func TestMiddlewareWrite(t *testing.T) { | |||||||
| 
 | 
 | ||||||
| 	w := performRequest(router, "GET", "/") | 	w := performRequest(router, "GET", "/") | ||||||
| 
 | 
 | ||||||
| 	assert.Equal(t, 400, w.Code) | 	assert.Equal(t, http.StatusBadRequest, w.Code) | ||||||
| 	assert.Equal(t, strings.Replace("hola\n<map><foo>bar</foo></map>{\"foo\":\"bar\"}{\"foo\":\"bar\"}event:test\ndata:message\n\n", " ", "", -1), strings.Replace(w.Body.String(), " ", "", -1)) | 	assert.Equal(t, strings.Replace("hola\n<map><foo>bar</foo></map>{\"foo\":\"bar\"}{\"foo\":\"bar\"}event:test\ndata:message\n\n", " ", "", -1), strings.Replace(w.Body.String(), " ", "", -1)) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										20
									
								
								mode.go
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								mode.go
									
									
									
									
									
								
							| @ -11,12 +11,16 @@ import ( | |||||||
| 	"github.com/gin-gonic/gin/binding" | 	"github.com/gin-gonic/gin/binding" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // ENV_GIN_MODE indicates environment name for gin mode. | ||||||
| const ENV_GIN_MODE = "GIN_MODE" | const ENV_GIN_MODE = "GIN_MODE" | ||||||
| 
 | 
 | ||||||
| const ( | const ( | ||||||
| 	DebugMode   = "debug" | 	// DebugMode indicates gin mode is debug. | ||||||
|  | 	DebugMode = "debug" | ||||||
|  | 	// ReleaseMode indicates gin mode is release. | ||||||
| 	ReleaseMode = "release" | 	ReleaseMode = "release" | ||||||
| 	TestMode    = "test" | 	// TestMode indicates gin mode is test. | ||||||
|  | 	TestMode = "test" | ||||||
| ) | ) | ||||||
| const ( | const ( | ||||||
| 	debugCode = iota | 	debugCode = iota | ||||||
| @ -24,7 +28,7 @@ const ( | |||||||
| 	testCode | 	testCode | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // DefaultWriter is the default io.Writer used the Gin for debug output and | // DefaultWriter is the default io.Writer used by Gin for debug output and | ||||||
| // middleware output like Logger() or Recovery(). | // middleware output like Logger() or Recovery(). | ||||||
| // Note that both Logger and Recovery provides custom ways to configure their | // Note that both Logger and Recovery provides custom ways to configure their | ||||||
| // output io.Writer. | // output io.Writer. | ||||||
| @ -32,6 +36,8 @@ const ( | |||||||
| // 		import "github.com/mattn/go-colorable" | // 		import "github.com/mattn/go-colorable" | ||||||
| // 		gin.DefaultWriter = colorable.NewColorableStdout() | // 		gin.DefaultWriter = colorable.NewColorableStdout() | ||||||
| var DefaultWriter io.Writer = os.Stdout | var DefaultWriter io.Writer = os.Stdout | ||||||
|  | 
 | ||||||
|  | // DefaultErrorWriter is the default io.Writer used by Gin to debug errors | ||||||
| var DefaultErrorWriter io.Writer = os.Stderr | var DefaultErrorWriter io.Writer = os.Stderr | ||||||
| 
 | 
 | ||||||
| var ginMode = debugCode | var ginMode = debugCode | ||||||
| @ -42,6 +48,7 @@ func init() { | |||||||
| 	SetMode(mode) | 	SetMode(mode) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // SetMode sets gin mode according to input string. | ||||||
| func SetMode(value string) { | func SetMode(value string) { | ||||||
| 	switch value { | 	switch value { | ||||||
| 	case DebugMode, "": | 	case DebugMode, "": | ||||||
| @ -53,17 +60,24 @@ func SetMode(value string) { | |||||||
| 	default: | 	default: | ||||||
| 		panic("gin mode unknown: " + value) | 		panic("gin mode unknown: " + value) | ||||||
| 	} | 	} | ||||||
|  | 	if value == "" { | ||||||
|  | 		value = DebugMode | ||||||
|  | 	} | ||||||
| 	modeName = value | 	modeName = value | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // DisableBindValidation closes the default validator. | ||||||
| func DisableBindValidation() { | func DisableBindValidation() { | ||||||
| 	binding.Validator = nil | 	binding.Validator = nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // EnableJsonDecoderUseNumber sets true for binding.EnableDecoderUseNumberto to | ||||||
|  | // call the UseNumber method on the JSON Decoder instance. | ||||||
| func EnableJsonDecoderUseNumber() { | func EnableJsonDecoderUseNumber() { | ||||||
| 	binding.EnableDecoderUseNumber = true | 	binding.EnableDecoderUseNumber = true | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Mode returns currently gin mode. | ||||||
| func Mode() string { | func Mode() string { | ||||||
| 	return modeName | 	return modeName | ||||||
| } | } | ||||||
|  | |||||||
| @ -21,6 +21,10 @@ func TestSetMode(t *testing.T) { | |||||||
| 	assert.Equal(t, TestMode, Mode()) | 	assert.Equal(t, TestMode, Mode()) | ||||||
| 	os.Unsetenv(ENV_GIN_MODE) | 	os.Unsetenv(ENV_GIN_MODE) | ||||||
| 
 | 
 | ||||||
|  | 	SetMode("") | ||||||
|  | 	assert.Equal(t, debugCode, ginMode) | ||||||
|  | 	assert.Equal(t, DebugMode, Mode()) | ||||||
|  | 
 | ||||||
| 	SetMode(DebugMode) | 	SetMode(DebugMode) | ||||||
| 	assert.Equal(t, debugCode, ginMode) | 	assert.Equal(t, debugCode, ginMode) | ||||||
| 	assert.Equal(t, DebugMode, Mode()) | 	assert.Equal(t, DebugMode, Mode()) | ||||||
|  | |||||||
							
								
								
									
										6
									
								
								path.go
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								path.go
									
									
									
									
									
								
							| @ -41,7 +41,7 @@ func cleanPath(p string) string { | |||||||
| 		buf[0] = '/' | 		buf[0] = '/' | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	trailing := n > 2 && p[n-1] == '/' | 	trailing := n > 1 && p[n-1] == '/' | ||||||
| 
 | 
 | ||||||
| 	// A bit more clunky without a 'lazybuf' like the path package, but the loop | 	// A bit more clunky without a 'lazybuf' like the path package, but the loop | ||||||
| 	// gets completely inlined (bufApp). So in contrast to the path package this | 	// gets completely inlined (bufApp). So in contrast to the path package this | ||||||
| @ -59,11 +59,11 @@ func cleanPath(p string) string { | |||||||
| 
 | 
 | ||||||
| 		case p[r] == '.' && p[r+1] == '/': | 		case p[r] == '.' && p[r+1] == '/': | ||||||
| 			// . element | 			// . element | ||||||
| 			r++ | 			r += 2 | ||||||
| 
 | 
 | ||||||
| 		case p[r] == '.' && p[r+1] == '.' && (r+2 == n || p[r+2] == '/'): | 		case p[r] == '.' && p[r+1] == '.' && (r+2 == n || p[r+2] == '/'): | ||||||
| 			// .. element: remove to last / | 			// .. element: remove to last / | ||||||
| 			r += 2 | 			r += 3 | ||||||
| 
 | 
 | ||||||
| 			if w > 1 { | 			if w > 1 { | ||||||
| 				// can backtrack | 				// can backtrack | ||||||
|  | |||||||
| @ -24,6 +24,7 @@ var cleanTests = []struct { | |||||||
| 
 | 
 | ||||||
| 	// missing root | 	// missing root | ||||||
| 	{"", "/"}, | 	{"", "/"}, | ||||||
|  | 	{"a/", "/a/"}, | ||||||
| 	{"abc", "/abc"}, | 	{"abc", "/abc"}, | ||||||
| 	{"abc/def", "/abc/def"}, | 	{"abc/def", "/abc/def"}, | ||||||
| 	{"a/b/c", "/a/b/c"}, | 	{"a/b/c", "/a/b/c"}, | ||||||
|  | |||||||
							
								
								
									
										39
									
								
								recovery.go
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								recovery.go
									
									
									
									
									
								
							| @ -10,8 +10,13 @@ import ( | |||||||
| 	"io" | 	"io" | ||||||
| 	"io/ioutil" | 	"io/ioutil" | ||||||
| 	"log" | 	"log" | ||||||
|  | 	"net" | ||||||
|  | 	"net/http" | ||||||
| 	"net/http/httputil" | 	"net/http/httputil" | ||||||
|  | 	"os" | ||||||
| 	"runtime" | 	"runtime" | ||||||
|  | 	"strings" | ||||||
|  | 	"time" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var ( | var ( | ||||||
| @ -35,12 +40,37 @@ func RecoveryWithWriter(out io.Writer) HandlerFunc { | |||||||
| 	return func(c *Context) { | 	return func(c *Context) { | ||||||
| 		defer func() { | 		defer func() { | ||||||
| 			if err := recover(); err != nil { | 			if err := recover(); err != nil { | ||||||
|  | 				// Check for a broken connection, as it is not really a | ||||||
|  | 				// condition that warrants a panic stack trace. | ||||||
|  | 				var brokenPipe bool | ||||||
|  | 				if ne, ok := err.(*net.OpError); ok { | ||||||
|  | 					if se, ok := ne.Err.(*os.SyscallError); ok { | ||||||
|  | 						if strings.Contains(strings.ToLower(se.Error()), "broken pipe") || strings.Contains(strings.ToLower(se.Error()), "connection reset by peer") { | ||||||
|  | 							brokenPipe = true | ||||||
|  | 						} | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
| 				if logger != nil { | 				if logger != nil { | ||||||
| 					stack := stack(3) | 					stack := stack(3) | ||||||
| 					httprequest, _ := httputil.DumpRequest(c.Request, false) | 					httprequest, _ := httputil.DumpRequest(c.Request, false) | ||||||
| 					logger.Printf("[Recovery] panic recovered:\n%s\n%s\n%s%s", string(httprequest), err, stack, reset) | 					if brokenPipe { | ||||||
|  | 						logger.Printf("%s\n%s%s", err, string(httprequest), reset) | ||||||
|  | 					} else if IsDebugging() { | ||||||
|  | 						logger.Printf("[Recovery] %s panic recovered:\n%s\n%s\n%s%s", | ||||||
|  | 							timeFormat(time.Now()), string(httprequest), err, stack, reset) | ||||||
|  | 					} else { | ||||||
|  | 						logger.Printf("[Recovery] %s panic recovered:\n%s\n%s%s", | ||||||
|  | 							timeFormat(time.Now()), err, stack, reset) | ||||||
|  | 					} | ||||||
|  | 				} | ||||||
|  | 
 | ||||||
|  | 				// If the connection is dead, we can't write a status to it. | ||||||
|  | 				if brokenPipe { | ||||||
|  | 					c.Error(err.(error)) // nolint: errcheck | ||||||
|  | 					c.Abort() | ||||||
|  | 				} else { | ||||||
|  | 					c.AbortWithStatus(http.StatusInternalServerError) | ||||||
| 				} | 				} | ||||||
| 				c.AbortWithStatus(500) |  | ||||||
| 			} | 			} | ||||||
| 		}() | 		}() | ||||||
| 		c.Next() | 		c.Next() | ||||||
| @ -107,3 +137,8 @@ func function(pc uintptr) []byte { | |||||||
| 	name = bytes.Replace(name, centerDot, dot, -1) | 	name = bytes.Replace(name, centerDot, dot, -1) | ||||||
| 	return name | 	return name | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | func timeFormat(t time.Time) string { | ||||||
|  | 	var timeString = t.Format("2006/01/02 - 15:04:05") | ||||||
|  | 	return timeString | ||||||
|  | } | ||||||
|  | |||||||
| @ -2,10 +2,17 @@ | |||||||
| // Use of this source code is governed by a MIT style | // Use of this source code is governed by a MIT style | ||||||
| // license that can be found in the LICENSE file. | // license that can be found in the LICENSE file. | ||||||
| 
 | 
 | ||||||
|  | // +build go1.7 | ||||||
|  | 
 | ||||||
| package gin | package gin | ||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"net" | ||||||
|  | 	"net/http" | ||||||
|  | 	"os" | ||||||
|  | 	"strings" | ||||||
|  | 	"syscall" | ||||||
| 	"testing" | 	"testing" | ||||||
| 
 | 
 | ||||||
| 	"github.com/stretchr/testify/assert" | 	"github.com/stretchr/testify/assert" | ||||||
| @ -22,10 +29,21 @@ func TestPanicInHandler(t *testing.T) { | |||||||
| 	// RUN | 	// RUN | ||||||
| 	w := performRequest(router, "GET", "/recovery") | 	w := performRequest(router, "GET", "/recovery") | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 500, w.Code) | 	assert.Equal(t, http.StatusInternalServerError, w.Code) | ||||||
| 	assert.Contains(t, buffer.String(), "GET /recovery") | 	assert.Contains(t, buffer.String(), "panic recovered") | ||||||
| 	assert.Contains(t, buffer.String(), "Oupps, Houston, we have a problem") | 	assert.Contains(t, buffer.String(), "Oupps, Houston, we have a problem") | ||||||
| 	assert.Contains(t, buffer.String(), "TestPanicInHandler") | 	assert.Contains(t, buffer.String(), "TestPanicInHandler") | ||||||
|  | 	assert.NotContains(t, buffer.String(), "GET /recovery") | ||||||
|  | 
 | ||||||
|  | 	// Debug mode prints the request | ||||||
|  | 	SetMode(DebugMode) | ||||||
|  | 	// RUN | ||||||
|  | 	w = performRequest(router, "GET", "/recovery") | ||||||
|  | 	// TEST | ||||||
|  | 	assert.Equal(t, http.StatusInternalServerError, w.Code) | ||||||
|  | 	assert.Contains(t, buffer.String(), "GET /recovery") | ||||||
|  | 
 | ||||||
|  | 	SetMode(TestMode) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // TestPanicWithAbort assert that panic has been recovered even if context.Abort was used. | // TestPanicWithAbort assert that panic has been recovered even if context.Abort was used. | ||||||
| @ -33,11 +51,66 @@ func TestPanicWithAbort(t *testing.T) { | |||||||
| 	router := New() | 	router := New() | ||||||
| 	router.Use(RecoveryWithWriter(nil)) | 	router.Use(RecoveryWithWriter(nil)) | ||||||
| 	router.GET("/recovery", func(c *Context) { | 	router.GET("/recovery", func(c *Context) { | ||||||
| 		c.AbortWithStatus(400) | 		c.AbortWithStatus(http.StatusBadRequest) | ||||||
| 		panic("Oupps, Houston, we have a problem") | 		panic("Oupps, Houston, we have a problem") | ||||||
| 	}) | 	}) | ||||||
| 	// RUN | 	// RUN | ||||||
| 	w := performRequest(router, "GET", "/recovery") | 	w := performRequest(router, "GET", "/recovery") | ||||||
| 	// TEST | 	// TEST | ||||||
| 	assert.Equal(t, 400, w.Code) | 	assert.Equal(t, http.StatusBadRequest, w.Code) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestSource(t *testing.T) { | ||||||
|  | 	bs := source(nil, 0) | ||||||
|  | 	assert.Equal(t, []byte("???"), bs) | ||||||
|  | 
 | ||||||
|  | 	in := [][]byte{ | ||||||
|  | 		[]byte("Hello world."), | ||||||
|  | 		[]byte("Hi, gin.."), | ||||||
|  | 	} | ||||||
|  | 	bs = source(in, 10) | ||||||
|  | 	assert.Equal(t, []byte("???"), bs) | ||||||
|  | 
 | ||||||
|  | 	bs = source(in, 1) | ||||||
|  | 	assert.Equal(t, []byte("Hello world."), bs) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | func TestFunction(t *testing.T) { | ||||||
|  | 	bs := function(1) | ||||||
|  | 	assert.Equal(t, []byte("???"), bs) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // TestPanicWithBrokenPipe asserts that recovery specifically handles | ||||||
|  | // writing responses to broken pipes | ||||||
|  | func TestPanicWithBrokenPipe(t *testing.T) { | ||||||
|  | 	const expectCode = 204 | ||||||
|  | 
 | ||||||
|  | 	expectMsgs := map[syscall.Errno]string{ | ||||||
|  | 		syscall.EPIPE:      "broken pipe", | ||||||
|  | 		syscall.ECONNRESET: "connection reset by peer", | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	for errno, expectMsg := range expectMsgs { | ||||||
|  | 		t.Run(expectMsg, func(t *testing.T) { | ||||||
|  | 
 | ||||||
|  | 			var buf bytes.Buffer | ||||||
|  | 
 | ||||||
|  | 			router := New() | ||||||
|  | 			router.Use(RecoveryWithWriter(&buf)) | ||||||
|  | 			router.GET("/recovery", func(c *Context) { | ||||||
|  | 				// Start writing response | ||||||
|  | 				c.Header("X-Test", "Value") | ||||||
|  | 				c.Status(expectCode) | ||||||
|  | 
 | ||||||
|  | 				// Oops. Client connection closed | ||||||
|  | 				e := &net.OpError{Err: &os.SyscallError{Err: errno}} | ||||||
|  | 				panic(e) | ||||||
|  | 			}) | ||||||
|  | 			// RUN | ||||||
|  | 			w := performRequest(router, "GET", "/recovery") | ||||||
|  | 			// TEST | ||||||
|  | 			assert.Equal(t, expectCode, w.Code) | ||||||
|  | 			assert.Contains(t, strings.ToLower(buf.String()), expectMsg) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
| } | } | ||||||
|  | |||||||
| @ -6,6 +6,7 @@ package render | |||||||
| 
 | 
 | ||||||
| import "net/http" | import "net/http" | ||||||
| 
 | 
 | ||||||
|  | // Data contains ContentType and bytes data. | ||||||
| type Data struct { | type Data struct { | ||||||
| 	ContentType string | 	ContentType string | ||||||
| 	Data        []byte | 	Data        []byte | ||||||
| @ -18,6 +19,7 @@ func (r Data) Render(w http.ResponseWriter) (err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (Data) writes custom ContentType. | ||||||
| func (r Data) WriteContentType(w http.ResponseWriter) { | func (r Data) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, []string{r.ContentType}) | 	writeContentType(w, []string{r.ContentType}) | ||||||
| } | } | ||||||
|  | |||||||
| @ -9,20 +9,27 @@ import ( | |||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Delims represents a set of Left and Right delimiters for HTML template rendering. | ||||||
| type Delims struct { | type Delims struct { | ||||||
| 	Left  string | 	// Left delimiter, defaults to {{. | ||||||
|  | 	Left string | ||||||
|  | 	// Right delimiter, defaults to }}. | ||||||
| 	Right string | 	Right string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // HTMLRender interface is to be implemented by HTMLProduction and HTMLDebug. | ||||||
| type HTMLRender interface { | type HTMLRender interface { | ||||||
|  | 	// Instance returns an HTML instance. | ||||||
| 	Instance(string, interface{}) Render | 	Instance(string, interface{}) Render | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // HTMLProduction contains template reference and its delims. | ||||||
| type HTMLProduction struct { | type HTMLProduction struct { | ||||||
| 	Template *template.Template | 	Template *template.Template | ||||||
| 	Delims   Delims | 	Delims   Delims | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // HTMLDebug contains template delims and pattern and function with file list. | ||||||
| type HTMLDebug struct { | type HTMLDebug struct { | ||||||
| 	Files   []string | 	Files   []string | ||||||
| 	Glob    string | 	Glob    string | ||||||
| @ -30,6 +37,7 @@ type HTMLDebug struct { | |||||||
| 	FuncMap template.FuncMap | 	FuncMap template.FuncMap | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // HTML contains template reference and its name with given interface object. | ||||||
| type HTML struct { | type HTML struct { | ||||||
| 	Template *template.Template | 	Template *template.Template | ||||||
| 	Name     string | 	Name     string | ||||||
| @ -38,6 +46,7 @@ type HTML struct { | |||||||
| 
 | 
 | ||||||
| var htmlContentType = []string{"text/html; charset=utf-8"} | var htmlContentType = []string{"text/html; charset=utf-8"} | ||||||
| 
 | 
 | ||||||
|  | // Instance (HTMLProduction) returns an HTML instance which it realizes Render interface. | ||||||
| func (r HTMLProduction) Instance(name string, data interface{}) Render { | func (r HTMLProduction) Instance(name string, data interface{}) Render { | ||||||
| 	return HTML{ | 	return HTML{ | ||||||
| 		Template: r.Template, | 		Template: r.Template, | ||||||
| @ -46,6 +55,7 @@ func (r HTMLProduction) Instance(name string, data interface{}) Render { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Instance (HTMLDebug) returns an HTML instance which it realizes Render interface. | ||||||
| func (r HTMLDebug) Instance(name string, data interface{}) Render { | func (r HTMLDebug) Instance(name string, data interface{}) Render { | ||||||
| 	return HTML{ | 	return HTML{ | ||||||
| 		Template: r.loadTemplate(), | 		Template: r.loadTemplate(), | ||||||
| @ -66,6 +76,7 @@ func (r HTMLDebug) loadTemplate() *template.Template { | |||||||
| 	panic("the HTML debug render was created without files or glob pattern") | 	panic("the HTML debug render was created without files or glob pattern") | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render (HTML) executes template and writes its result with custom ContentType for response. | ||||||
| func (r HTML) Render(w http.ResponseWriter) error { | func (r HTML) Render(w http.ResponseWriter) error { | ||||||
| 	r.WriteContentType(w) | 	r.WriteContentType(w) | ||||||
| 
 | 
 | ||||||
| @ -75,6 +86,7 @@ func (r HTML) Render(w http.ResponseWriter) error { | |||||||
| 	return r.Template.ExecuteTemplate(w, r.Name, r.Data) | 	return r.Template.ExecuteTemplate(w, r.Name, r.Data) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (HTML) writes HTML ContentType. | ||||||
| func (r HTML) WriteContentType(w http.ResponseWriter) { | func (r HTML) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, htmlContentType) | 	writeContentType(w, htmlContentType) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										110
									
								
								render/json.go
									
									
									
									
									
								
							
							
						
						
									
										110
									
								
								render/json.go
									
									
									
									
									
								
							| @ -6,28 +6,48 @@ package render | |||||||
| 
 | 
 | ||||||
| import ( | import ( | ||||||
| 	"bytes" | 	"bytes" | ||||||
|  | 	"fmt" | ||||||
|  | 	"html/template" | ||||||
| 	"net/http" | 	"net/http" | ||||||
| 
 | 
 | ||||||
| 	"github.com/gin-gonic/gin/json" | 	"github.com/gin-gonic/gin/internal/json" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // JSON contains the given interface object. | ||||||
| type JSON struct { | type JSON struct { | ||||||
| 	Data interface{} | 	Data interface{} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // IndentedJSON contains the given interface object. | ||||||
| type IndentedJSON struct { | type IndentedJSON struct { | ||||||
| 	Data interface{} | 	Data interface{} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // SecureJSON contains the given interface object and its prefix. | ||||||
| type SecureJSON struct { | type SecureJSON struct { | ||||||
| 	Prefix string | 	Prefix string | ||||||
| 	Data   interface{} | 	Data   interface{} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // JsonpJSON contains the given interface object its callback. | ||||||
|  | type JsonpJSON struct { | ||||||
|  | 	Callback string | ||||||
|  | 	Data     interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // AsciiJSON contains the given interface object. | ||||||
|  | type AsciiJSON struct { | ||||||
|  | 	Data interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // SecureJSONPrefix is a string which represents SecureJSON prefix. | ||||||
| type SecureJSONPrefix string | type SecureJSONPrefix string | ||||||
| 
 | 
 | ||||||
| var jsonContentType = []string{"application/json; charset=utf-8"} | var jsonContentType = []string{"application/json; charset=utf-8"} | ||||||
|  | var jsonpContentType = []string{"application/javascript; charset=utf-8"} | ||||||
|  | var jsonAsciiContentType = []string{"application/json"} | ||||||
| 
 | 
 | ||||||
|  | // Render (JSON) writes data with custom ContentType. | ||||||
| func (r JSON) Render(w http.ResponseWriter) (err error) { | func (r JSON) Render(w http.ResponseWriter) (err error) { | ||||||
| 	if err = WriteJSON(w, r.Data); err != nil { | 	if err = WriteJSON(w, r.Data); err != nil { | ||||||
| 		panic(err) | 		panic(err) | ||||||
| @ -35,34 +55,39 @@ func (r JSON) Render(w http.ResponseWriter) (err error) { | |||||||
| 	return | 	return | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (JSON) writes JSON ContentType. | ||||||
| func (r JSON) WriteContentType(w http.ResponseWriter) { | func (r JSON) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, jsonContentType) | 	writeContentType(w, jsonContentType) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteJSON marshals the given interface object and writes it with custom ContentType. | ||||||
| func WriteJSON(w http.ResponseWriter, obj interface{}) error { | func WriteJSON(w http.ResponseWriter, obj interface{}) error { | ||||||
| 	writeContentType(w, jsonContentType) | 	writeContentType(w, jsonContentType) | ||||||
| 	jsonBytes, err := json.Marshal(obj) | 	jsonBytes, err := json.Marshal(obj) | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	w.Write(jsonBytes) | 	_, err = w.Write(jsonBytes) | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render (IndentedJSON) marshals the given interface object and writes it with custom ContentType. | ||||||
| func (r IndentedJSON) Render(w http.ResponseWriter) error { | func (r IndentedJSON) Render(w http.ResponseWriter) error { | ||||||
| 	r.WriteContentType(w) | 	r.WriteContentType(w) | ||||||
| 	jsonBytes, err := json.MarshalIndent(r.Data, "", "    ") | 	jsonBytes, err := json.MarshalIndent(r.Data, "", "    ") | ||||||
| 	if err != nil { | 	if err != nil { | ||||||
| 		return err | 		return err | ||||||
| 	} | 	} | ||||||
| 	w.Write(jsonBytes) | 	_, err = w.Write(jsonBytes) | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (IndentedJSON) writes JSON ContentType. | ||||||
| func (r IndentedJSON) WriteContentType(w http.ResponseWriter) { | func (r IndentedJSON) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, jsonContentType) | 	writeContentType(w, jsonContentType) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render (SecureJSON) marshals the given interface object and writes it with custom ContentType. | ||||||
| func (r SecureJSON) Render(w http.ResponseWriter) error { | func (r SecureJSON) Render(w http.ResponseWriter) error { | ||||||
| 	r.WriteContentType(w) | 	r.WriteContentType(w) | ||||||
| 	jsonBytes, err := json.Marshal(r.Data) | 	jsonBytes, err := json.Marshal(r.Data) | ||||||
| @ -71,12 +96,81 @@ func (r SecureJSON) Render(w http.ResponseWriter) error { | |||||||
| 	} | 	} | ||||||
| 	// if the jsonBytes is array values | 	// if the jsonBytes is array values | ||||||
| 	if bytes.HasPrefix(jsonBytes, []byte("[")) && bytes.HasSuffix(jsonBytes, []byte("]")) { | 	if bytes.HasPrefix(jsonBytes, []byte("[")) && bytes.HasSuffix(jsonBytes, []byte("]")) { | ||||||
| 		w.Write([]byte(r.Prefix)) | 		_, err = w.Write([]byte(r.Prefix)) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return err | ||||||
|  | 		} | ||||||
| 	} | 	} | ||||||
| 	w.Write(jsonBytes) | 	_, err = w.Write(jsonBytes) | ||||||
| 	return nil | 	return err | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (SecureJSON) writes JSON ContentType. | ||||||
| func (r SecureJSON) WriteContentType(w http.ResponseWriter) { | func (r SecureJSON) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, jsonContentType) | 	writeContentType(w, jsonContentType) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // Render (JsonpJSON) marshals the given interface object and writes it and its callback with custom ContentType. | ||||||
|  | func (r JsonpJSON) Render(w http.ResponseWriter) (err error) { | ||||||
|  | 	r.WriteContentType(w) | ||||||
|  | 	ret, err := json.Marshal(r.Data) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if r.Callback == "" { | ||||||
|  | 		_, err = w.Write(ret) | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	callback := template.JSEscapeString(r.Callback) | ||||||
|  | 	_, err = w.Write([]byte(callback)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	_, err = w.Write([]byte("(")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	_, err = w.Write(ret) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 	_, err = w.Write([]byte(")")) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WriteContentType (JsonpJSON) writes Javascript ContentType. | ||||||
|  | func (r JsonpJSON) WriteContentType(w http.ResponseWriter) { | ||||||
|  | 	writeContentType(w, jsonpContentType) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Render (AsciiJSON) marshals the given interface object and writes it with custom ContentType. | ||||||
|  | func (r AsciiJSON) Render(w http.ResponseWriter) (err error) { | ||||||
|  | 	r.WriteContentType(w) | ||||||
|  | 	ret, err := json.Marshal(r.Data) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	var buffer bytes.Buffer | ||||||
|  | 	for _, r := range string(ret) { | ||||||
|  | 		cvt := string(r) | ||||||
|  | 		if r >= 128 { | ||||||
|  | 			cvt = fmt.Sprintf("\\u%04x", int64(r)) | ||||||
|  | 		} | ||||||
|  | 		buffer.WriteString(cvt) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	_, err = w.Write(buffer.Bytes()) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WriteContentType (AsciiJSON) writes JSON ContentType. | ||||||
|  | func (r AsciiJSON) WriteContentType(w http.ResponseWriter) { | ||||||
|  | 	writeContentType(w, jsonAsciiContentType) | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										31
									
								
								render/json_17.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								render/json_17.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build go1.7 | ||||||
|  | 
 | ||||||
|  | package render | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
|  | 	"github.com/gin-gonic/gin/internal/json" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // PureJSON contains the given interface object. | ||||||
|  | type PureJSON struct { | ||||||
|  | 	Data interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Render (PureJSON) writes custom ContentType and encodes the given interface object. | ||||||
|  | func (r PureJSON) Render(w http.ResponseWriter) error { | ||||||
|  | 	r.WriteContentType(w) | ||||||
|  | 	encoder := json.NewEncoder(w) | ||||||
|  | 	encoder.SetEscapeHTML(false) | ||||||
|  | 	return encoder.Encode(r.Data) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WriteContentType (PureJSON) writes custom ContentType. | ||||||
|  | func (r PureJSON) WriteContentType(w http.ResponseWriter) { | ||||||
|  | 	writeContentType(w, jsonContentType) | ||||||
|  | } | ||||||
| @ -10,22 +10,26 @@ import ( | |||||||
| 	"github.com/ugorji/go/codec" | 	"github.com/ugorji/go/codec" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // MsgPack contains the given interface object. | ||||||
| type MsgPack struct { | type MsgPack struct { | ||||||
| 	Data interface{} | 	Data interface{} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var msgpackContentType = []string{"application/msgpack; charset=utf-8"} | var msgpackContentType = []string{"application/msgpack; charset=utf-8"} | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (MsgPack) writes MsgPack ContentType. | ||||||
| func (r MsgPack) WriteContentType(w http.ResponseWriter) { | func (r MsgPack) WriteContentType(w http.ResponseWriter) { | ||||||
| 	writeContentType(w, msgpackContentType) | 	writeContentType(w, msgpackContentType) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render (MsgPack) encodes the given interface object and writes data with custom ContentType. | ||||||
| func (r MsgPack) Render(w http.ResponseWriter) error { | func (r MsgPack) Render(w http.ResponseWriter) error { | ||||||
| 	return WriteMsgPack(w, r.Data) | 	return WriteMsgPack(w, r.Data) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteMsgPack writes MsgPack ContentType and encodes the given interface object. | ||||||
| func WriteMsgPack(w http.ResponseWriter, obj interface{}) error { | func WriteMsgPack(w http.ResponseWriter, obj interface{}) error { | ||||||
| 	writeContentType(w, msgpackContentType) | 	writeContentType(w, msgpackContentType) | ||||||
| 	var h codec.Handle = new(codec.MsgpackHandle) | 	var mh codec.MsgpackHandle | ||||||
| 	return codec.NewEncoder(w, h).Encode(obj) | 	return codec.NewEncoder(w, &mh).Encode(obj) | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										36
									
								
								render/protobuf.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								render/protobuf.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package render | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"net/http" | ||||||
|  | 
 | ||||||
|  | 	"github.com/golang/protobuf/proto" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // ProtoBuf contains the given interface object. | ||||||
|  | type ProtoBuf struct { | ||||||
|  | 	Data interface{} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | var protobufContentType = []string{"application/x-protobuf"} | ||||||
|  | 
 | ||||||
|  | // Render (ProtoBuf) marshals the given interface object and writes data with custom ContentType. | ||||||
|  | func (r ProtoBuf) Render(w http.ResponseWriter) error { | ||||||
|  | 	r.WriteContentType(w) | ||||||
|  | 
 | ||||||
|  | 	bytes, err := proto.Marshal(r.Data.(proto.Message)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return err | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	_, err = w.Write(bytes) | ||||||
|  | 	return err | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WriteContentType (ProtoBuf) writes ProtoBuf ContentType. | ||||||
|  | func (r ProtoBuf) WriteContentType(w http.ResponseWriter) { | ||||||
|  | 	writeContentType(w, protobufContentType) | ||||||
|  | } | ||||||
							
								
								
									
										43
									
								
								render/reader.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								render/reader.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | package render | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"io" | ||||||
|  | 	"net/http" | ||||||
|  | 	"strconv" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | // Reader contains the IO reader and its length, and custom ContentType and other headers. | ||||||
|  | type Reader struct { | ||||||
|  | 	ContentType   string | ||||||
|  | 	ContentLength int64 | ||||||
|  | 	Reader        io.Reader | ||||||
|  | 	Headers       map[string]string | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // Render (Reader) writes data with custom ContentType and headers. | ||||||
|  | func (r Reader) Render(w http.ResponseWriter) (err error) { | ||||||
|  | 	r.WriteContentType(w) | ||||||
|  | 	r.Headers["Content-Length"] = strconv.FormatInt(r.ContentLength, 10) | ||||||
|  | 	r.writeHeaders(w, r.Headers) | ||||||
|  | 	_, err = io.Copy(w, r.Reader) | ||||||
|  | 	return | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // WriteContentType (Reader) writes custom ContentType. | ||||||
|  | func (r Reader) WriteContentType(w http.ResponseWriter) { | ||||||
|  | 	writeContentType(w, []string{r.ContentType}) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // writeHeaders writes custom Header. | ||||||
|  | func (r Reader) writeHeaders(w http.ResponseWriter, headers map[string]string) { | ||||||
|  | 	header := w.Header() | ||||||
|  | 	for k, v := range headers { | ||||||
|  | 		if val := header[k]; len(val) == 0 { | ||||||
|  | 			header[k] = []string{v} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -9,13 +9,17 @@ import ( | |||||||
| 	"net/http" | 	"net/http" | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
|  | // Redirect contains the http request reference and redirects status code and location. | ||||||
| type Redirect struct { | type Redirect struct { | ||||||
| 	Code     int | 	Code     int | ||||||
| 	Request  *http.Request | 	Request  *http.Request | ||||||
| 	Location string | 	Location string | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // Render (Redirect) redirects the http request to new location and writes redirect response. | ||||||
| func (r Redirect) Render(w http.ResponseWriter) error { | func (r Redirect) Render(w http.ResponseWriter) error { | ||||||
|  | 	// todo(thinkerou): go1.6 not support StatusPermanentRedirect(308) | ||||||
|  | 	// when we upgrade go version we can use http.StatusPermanentRedirect | ||||||
| 	if (r.Code < 300 || r.Code > 308) && r.Code != 201 { | 	if (r.Code < 300 || r.Code > 308) && r.Code != 201 { | ||||||
| 		panic(fmt.Sprintf("Cannot redirect with status code %d", r.Code)) | 		panic(fmt.Sprintf("Cannot redirect with status code %d", r.Code)) | ||||||
| 	} | 	} | ||||||
| @ -23,4 +27,5 @@ func (r Redirect) Render(w http.ResponseWriter) error { | |||||||
| 	return nil | 	return nil | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | // WriteContentType (Redirect) don't write any ContentType. | ||||||
| func (r Redirect) WriteContentType(http.ResponseWriter) {} | func (r Redirect) WriteContentType(http.ResponseWriter) {} | ||||||
|  | |||||||
| @ -6,8 +6,11 @@ package render | |||||||
| 
 | 
 | ||||||
| import "net/http" | import "net/http" | ||||||
| 
 | 
 | ||||||
|  | // Render interface is to be implemented by JSON, XML, HTML, YAML and so on. | ||||||
| type Render interface { | type Render interface { | ||||||
|  | 	// Render writes data with custom ContentType. | ||||||
| 	Render(http.ResponseWriter) error | 	Render(http.ResponseWriter) error | ||||||
|  | 	// WriteContentType writes custom ContentType. | ||||||
| 	WriteContentType(w http.ResponseWriter) | 	WriteContentType(w http.ResponseWriter) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -15,6 +18,7 @@ var ( | |||||||
| 	_ Render     = JSON{} | 	_ Render     = JSON{} | ||||||
| 	_ Render     = IndentedJSON{} | 	_ Render     = IndentedJSON{} | ||||||
| 	_ Render     = SecureJSON{} | 	_ Render     = SecureJSON{} | ||||||
|  | 	_ Render     = JsonpJSON{} | ||||||
| 	_ Render     = XML{} | 	_ Render     = XML{} | ||||||
| 	_ Render     = String{} | 	_ Render     = String{} | ||||||
| 	_ Render     = Redirect{} | 	_ Render     = Redirect{} | ||||||
| @ -24,6 +28,9 @@ var ( | |||||||
| 	_ HTMLRender = HTMLProduction{} | 	_ HTMLRender = HTMLProduction{} | ||||||
| 	_ Render     = YAML{} | 	_ Render     = YAML{} | ||||||
| 	_ Render     = MsgPack{} | 	_ Render     = MsgPack{} | ||||||
|  | 	_ Render     = Reader{} | ||||||
|  | 	_ Render     = AsciiJSON{} | ||||||
|  | 	_ Render     = ProtoBuf{} | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| func writeContentType(w http.ResponseWriter, value []string) { | func writeContentType(w http.ResponseWriter, value []string) { | ||||||
|  | |||||||
							
								
								
									
										26
									
								
								render/render_17_test.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								render/render_17_test.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | |||||||
|  | // Copyright 2018 Gin Core Team.  All rights reserved. | ||||||
|  | // Use of this source code is governed by a MIT style | ||||||
|  | // license that can be found in the LICENSE file. | ||||||
|  | 
 | ||||||
|  | // +build go1.7 | ||||||
|  | 
 | ||||||
|  | package render | ||||||
|  | 
 | ||||||
|  | import ( | ||||||
|  | 	"net/http/httptest" | ||||||
|  | 	"testing" | ||||||
|  | 
 | ||||||
|  | 	"github.com/stretchr/testify/assert" | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | func TestRenderPureJSON(t *testing.T) { | ||||||
|  | 	w := httptest.NewRecorder() | ||||||
|  | 	data := map[string]interface{}{ | ||||||
|  | 		"foo":  "bar", | ||||||
|  | 		"html": "<b>", | ||||||
|  | 	} | ||||||
|  | 	err := (PureJSON{data}).Render(w) | ||||||
|  | 	assert.NoError(t, err) | ||||||
|  | 	assert.Equal(t, "{\"foo\":\"bar\",\"html\":\"<b>\"}\n", w.Body.String()) | ||||||
|  | 	assert.Equal(t, "application/json; charset=utf-8", w.Header().Get("Content-Type")) | ||||||
|  | } | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user