diff --git a/.travis.yml b/.travis.yml
index 27c80ef8..748a07a7 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -8,8 +8,8 @@ matrix:
env: GO111MODULE=on
- go: 1.12.x
env: GO111MODULE=on
+ - go: 1.13.x
- go: master
- env: GO111MODULE=on
git:
depth: 10
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 15dfb1a8..6ccd2faf 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -15,7 +15,7 @@
- [NEW] Refactor form mappings [#1749](https://github.com/gin-gonic/gin/pull/1749)
- [NEW] Added flag to context.Stream indicates if client disconnected in middle of stream [#1252](https://github.com/gin-gonic/gin/pull/1252)
- [FIX] Moved [examples](https://github.com/gin-gonic/examples) to stand alone Repo [#1775](https://github.com/gin-gonic/gin/pull/1775)
-- [NEW] Extend context.File to allow for the content-dispositon attachments via a new method context.Attachment [#1260](https://github.com/gin-gonic/gin/pull/1260)
+- [NEW] Extend context.File to allow for the content-disposition attachments via a new method context.Attachment [#1260](https://github.com/gin-gonic/gin/pull/1260)
- [FIX] Support HTTP content negotiation wildcards [#1112](https://github.com/gin-gonic/gin/pull/1112)
- [NEW] Add prefix from X-Forwarded-Prefix in redirectTrailingSlash [#1238](https://github.com/gin-gonic/gin/pull/1238)
- [FIX] context.Copy() race condition [#1020](https://github.com/gin-gonic/gin/pull/1020)
@@ -231,7 +231,7 @@
- [PERFORMANCE] Improve context's memory locality, reduce CPU cache faults.
- [NEW] Flexible rendering API
- [NEW] Add Context.File()
-- [NEW] Add shorcut RunTLS() for http.ListenAndServeTLS
+- [NEW] Add shortcut RunTLS() for http.ListenAndServeTLS
- [FIX] Rename NotFound404() to NoRoute()
- [FIX] Errors in context are purged
- [FIX] Adds HEAD method in Static file serving
@@ -254,7 +254,7 @@
- [NEW] New Bind() and BindWith() methods for parsing request body.
- [NEW] Add Content.Copy()
- [NEW] Add context.LastError()
-- [NEW] Add shorcut for OPTIONS HTTP method
+- [NEW] Add shortcut for OPTIONS HTTP method
- [FIX] Tons of README fixes
- [FIX] Header is written before body
- [FIX] BasicAuth() and changes API a little bit
diff --git a/README.md b/README.md
index 2761259e..22f83b65 100644
--- a/README.md
+++ b/README.md
@@ -101,6 +101,12 @@ $ go get github.com/kardianos/govendor
$ mkdir -p $GOPATH/src/github.com/myusername/project && cd "$_"
```
+If you are on a Mac and you're installing Go 1.8 (released: Feb 2017) or later, GOPATH is automatically determined by the Go toolchain for you. It defaults to $HOME/go on macOS so you can create your project like this
+
+```sh
+$ mkdir -p $HOME/go/src/github.com/myusername/project && cd "$_"
+```
+
3. Vendor init your project and add gin
```sh
@@ -622,10 +628,10 @@ Note that you need to set the corresponding binding tag on all fields you want t
Also, Gin provides two sets of methods for binding:
- **Type** - Must bind
- - **Methods** - `Bind`, `BindJSON`, `BindXML`, `BindQuery`, `BindYAML`
+ - **Methods** - `Bind`, `BindJSON`, `BindXML`, `BindQuery`, `BindYAML`, `BindHeader`
- **Behavior** - These methods use `MustBindWith` under the hood. If there is a binding error, the request is aborted with `c.AbortWithError(400, err).SetType(ErrorTypeBind)`. This sets the response status code to 400 and the `Content-Type` header is set to `text/plain; charset=utf-8`. Note that if you try to set the response code after this, it will result in a warning `[GIN-debug] [WARNING] Headers were already written. Wanted to override status code 400 with 422`. If you wish to have greater control over the behavior, consider using the `ShouldBind` equivalent method.
- **Type** - Should bind
- - **Methods** - `ShouldBind`, `ShouldBindJSON`, `ShouldBindXML`, `ShouldBindQuery`, `ShouldBindYAML`
+ - **Methods** - `ShouldBind`, `ShouldBindJSON`, `ShouldBindXML`, `ShouldBindQuery`, `ShouldBindYAML`, `ShouldBindHeader`
- **Behavior** - These methods use `ShouldBindWith` under the hood. If there is a binding error, the error is returned and it is the developer's responsibility to handle the request and error appropriately.
When using the Bind-method, Gin tries to infer the binder depending on the Content-Type header. If you are sure what you are binding, you can use `MustBindWith` or `ShouldBindWith`.
@@ -1143,7 +1149,7 @@ func main() {
#### AsciiJSON
-Using AsciiJSON to Generates ASCII-only JSON with escaped non-ASCII chracters.
+Using AsciiJSON to Generates ASCII-only JSON with escaped non-ASCII characters.
```go
func main() {
@@ -1672,11 +1678,19 @@ func main() {
}
g.Go(func() error {
- return server01.ListenAndServe()
+ err := server01.ListenAndServe()
+ if err != nil && err != http.ErrServerClosed {
+ log.Fatal(err)
+ }
+ return err
})
g.Go(func() error {
- return server02.ListenAndServe()
+ err := server02.ListenAndServe()
+ if err != nil && err != http.ErrServerClosed {
+ log.Fatal(err)
+ }
+ return err
})
if err := g.Wait(); err != nil {
@@ -1793,6 +1807,7 @@ func main() {
func loadTemplate() (*template.Template, error) {
t := template.New("")
for name, file := range Assets.Files {
+ defer file.Close()
if file.IsDir() || !strings.HasSuffix(name, ".tmpl") {
continue
}
diff --git a/auth.go b/auth.go
index 9ed81b5d..c96b1e29 100644
--- a/auth.go
+++ b/auth.go
@@ -5,7 +5,6 @@
package gin
import (
- "crypto/subtle"
"encoding/base64"
"net/http"
"strconv"
@@ -86,11 +85,3 @@ func authorizationHeader(user, password string) string {
base := user + ":" + password
return "Basic " + base64.StdEncoding.EncodeToString([]byte(base))
}
-
-func secureCompare(given, actual string) bool {
- if subtle.ConstantTimeEq(int32(len(given)), int32(len(actual))) == 1 {
- return subtle.ConstantTimeCompare([]byte(given), []byte(actual)) == 1
- }
- // Securely compare actual to itself to keep constant time, but always return false.
- return subtle.ConstantTimeCompare([]byte(actual), []byte(actual)) == 1 && false
-}
diff --git a/auth_test.go b/auth_test.go
index 197e9208..e44bd100 100644
--- a/auth_test.go
+++ b/auth_test.go
@@ -81,13 +81,6 @@ func TestBasicAuthAuthorizationHeader(t *testing.T) {
assert.Equal(t, "Basic YWRtaW46cGFzc3dvcmQ=", authorizationHeader("admin", "password"))
}
-func TestBasicAuthSecureCompare(t *testing.T) {
- assert.True(t, secureCompare("1234567890", "1234567890"))
- assert.False(t, secureCompare("123456789", "1234567890"))
- assert.False(t, secureCompare("12345678900", "1234567890"))
- assert.False(t, secureCompare("1234567891", "1234567890"))
-}
-
func TestBasicAuthSucceed(t *testing.T) {
accounts := Accounts{"admin": "password"}
router := New()
diff --git a/binding/binding_body_test.go b/binding/binding_body_test.go
deleted file mode 100644
index 901d429c..00000000
--- a/binding/binding_body_test.go
+++ /dev/null
@@ -1,72 +0,0 @@
-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: `
-
- FOO
-`,
- },
- {
- 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)
-}
diff --git a/binding/binding_test.go b/binding/binding_test.go
index 806f3ac9..caabaace 100644
--- a/binding/binding_test.go
+++ b/binding/binding_test.go
@@ -64,6 +64,10 @@ type FooStructUseNumber struct {
Foo interface{} `json:"foo" binding:"required"`
}
+type FooStructDisallowUnknownFields struct {
+ Foo interface{} `json:"foo" binding:"required"`
+}
+
type FooBarStructForTimeType struct {
TimeFoo time.Time `form:"time_foo" time_format:"2006-01-02" time_utc:"1" time_location:"Asia/Chongqing"`
TimeBar time.Time `form:"time_bar" time_format:"2006-01-02" time_utc:"1"`
@@ -194,6 +198,12 @@ func TestBindingJSONUseNumber2(t *testing.T) {
`{"foo": 123}`, `{"bar": "foo"}`)
}
+func TestBindingJSONDisallowUnknownFields(t *testing.T) {
+ testBodyBindingDisallowUnknownFields(t, JSON,
+ "/", "/",
+ `{"foo": "bar"}`, `{"foo": "bar", "what": "this"}`)
+}
+
func TestBindingForm(t *testing.T) {
testFormBinding(t, "POST",
"/", "/",
@@ -658,9 +668,9 @@ func TestValidationDisabled(t *testing.T) {
assert.NoError(t, err)
}
-func TestExistsSucceeds(t *testing.T) {
+func TestRequiredSucceeds(t *testing.T) {
type HogeStruct struct {
- Hoge *int `json:"hoge" binding:"exists"`
+ Hoge *int `json:"hoge" binding:"required"`
}
var obj HogeStruct
@@ -669,9 +679,9 @@ func TestExistsSucceeds(t *testing.T) {
assert.NoError(t, err)
}
-func TestExistsFails(t *testing.T) {
+func TestRequiredFails(t *testing.T) {
type HogeStruct struct {
- Hoge *int `json:"foo" binding:"exists"`
+ Hoge *int `json:"foo" binding:"required"`
}
var obj HogeStruct
@@ -1162,6 +1172,25 @@ func testBodyBindingUseNumber2(t *testing.T, b Binding, name, path, badPath, bod
assert.Error(t, err)
}
+func testBodyBindingDisallowUnknownFields(t *testing.T, b Binding, path, badPath, body, badBody string) {
+ EnableDecoderDisallowUnknownFields = true
+ defer func() {
+ EnableDecoderDisallowUnknownFields = false
+ }()
+
+ obj := FooStructDisallowUnknownFields{}
+ req := requestWithBody("POST", path, body)
+ err := b.Bind(req, &obj)
+ assert.NoError(t, err)
+ assert.Equal(t, "bar", obj.Foo)
+
+ obj = FooStructDisallowUnknownFields{}
+ req = requestWithBody("POST", badPath, badBody)
+ err = JSON.Bind(req, &obj)
+ assert.Error(t, err)
+ assert.Contains(t, err.Error(), "what")
+}
+
func testBodyBindingFail(t *testing.T, b Binding, name, path, badPath, body, badBody string) {
assert.Equal(t, name, b.Name())
diff --git a/binding/default_validator.go b/binding/default_validator.go
index e7a302de..50e0d57c 100644
--- a/binding/default_validator.go
+++ b/binding/default_validator.go
@@ -8,7 +8,7 @@ import (
"reflect"
"sync"
- "gopkg.in/go-playground/validator.v8"
+ "gopkg.in/go-playground/validator.v9"
)
type defaultValidator struct {
@@ -45,7 +45,7 @@ func (v *defaultValidator) Engine() interface{} {
func (v *defaultValidator) lazyinit() {
v.once.Do(func() {
- config := &validator.Config{TagName: "binding"}
- v.validate = validator.New(config)
+ v.validate = validator.New()
+ v.validate.SetTagName("binding")
})
}
diff --git a/binding/json.go b/binding/json.go
index f968161b..d62e0705 100644
--- a/binding/json.go
+++ b/binding/json.go
@@ -18,6 +18,12 @@ import (
// interface{} as a Number instead of as a float64.
var EnableDecoderUseNumber = false
+// EnableDecoderDisallowUnknownFields is used to call the DisallowUnknownFields method
+// on the JSON Decoder instance. DisallowUnknownFields causes the Decoder to
+// return an error when the destination is a struct and the input contains object
+// keys which do not match any non-ignored, exported fields in the destination.
+var EnableDecoderDisallowUnknownFields = false
+
type jsonBinding struct{}
func (jsonBinding) Name() string {
@@ -40,6 +46,9 @@ func decodeJSON(r io.Reader, obj interface{}) error {
if EnableDecoderUseNumber {
decoder.UseNumber()
}
+ if EnableDecoderDisallowUnknownFields {
+ decoder.DisallowUnknownFields()
+ }
if err := decoder.Decode(obj); err != nil {
return err
}
diff --git a/binding/json_test.go b/binding/json_test.go
new file mode 100644
index 00000000..cae4cccc
--- /dev/null
+++ b/binding/json_test.go
@@ -0,0 +1,21 @@
+// Copyright 2019 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 (
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+)
+
+func TestJSONBindingBindBody(t *testing.T) {
+ var s struct {
+ Foo string `json:"foo"`
+ }
+ err := jsonBinding{}.BindBody([]byte(`{"foo": "FOO"}`), &s)
+ require.NoError(t, err)
+ assert.Equal(t, "FOO", s.Foo)
+}
diff --git a/binding/msgpack_test.go b/binding/msgpack_test.go
new file mode 100644
index 00000000..6baa6739
--- /dev/null
+++ b/binding/msgpack_test.go
@@ -0,0 +1,32 @@
+// Copyright 2019 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"
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+ "github.com/ugorji/go/codec"
+)
+
+func TestMsgpackBindingBindBody(t *testing.T) {
+ type teststruct struct {
+ Foo string `msgpack:"foo"`
+ }
+ var s teststruct
+ err := msgpackBinding{}.BindBody(msgpackBody(t, teststruct{"FOO"}), &s)
+ require.NoError(t, err)
+ assert.Equal(t, "FOO", s.Foo)
+}
+
+func msgpackBody(t *testing.T, obj interface{}) []byte {
+ var bs bytes.Buffer
+ h := &codec.MsgpackHandle{}
+ err := codec.NewEncoder(&bs, h).Encode(obj)
+ require.NoError(t, err)
+ return bs.Bytes()
+}
diff --git a/binding/validate_test.go b/binding/validate_test.go
index 2c76b6d6..81f78834 100644
--- a/binding/validate_test.go
+++ b/binding/validate_test.go
@@ -6,12 +6,11 @@ package binding
import (
"bytes"
- "reflect"
"testing"
"time"
"github.com/stretchr/testify/assert"
- "gopkg.in/go-playground/validator.v8"
+ "gopkg.in/go-playground/validator.v9"
)
type testInterface interface {
@@ -200,15 +199,8 @@ type structCustomValidation struct {
Integer int `binding:"notone"`
}
-// notOne is a custom validator meant to be used with `validator.v8` library.
-// The method signature for `v9` is significantly different and this function
-// would need to be changed for tests to pass after upgrade.
-// See https://github.com/gin-gonic/gin/pull/1015.
-func notOne(
- v *validator.Validate, topStruct reflect.Value, currentStructOrField reflect.Value,
- field reflect.Value, fieldType reflect.Type, fieldKind reflect.Kind, param string,
-) bool {
- if val, ok := field.Interface().(int); ok {
+func notOne(f1 validator.FieldLevel) bool {
+ if val, ok := f1.Field().Interface().(int); ok {
return val != 1
}
return false
diff --git a/binding/xml_test.go b/binding/xml_test.go
new file mode 100644
index 00000000..f9546c1a
--- /dev/null
+++ b/binding/xml_test.go
@@ -0,0 +1,25 @@
+// Copyright 2019 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 (
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+)
+
+func TestXMLBindingBindBody(t *testing.T) {
+ var s struct {
+ Foo string `xml:"foo"`
+ }
+ xmlBody := `
+
+ FOO
+`
+ err := xmlBinding{}.BindBody([]byte(xmlBody), &s)
+ require.NoError(t, err)
+ assert.Equal(t, "FOO", s.Foo)
+}
diff --git a/binding/yaml_test.go b/binding/yaml_test.go
new file mode 100644
index 00000000..e66338b7
--- /dev/null
+++ b/binding/yaml_test.go
@@ -0,0 +1,21 @@
+// Copyright 2019 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 (
+ "testing"
+
+ "github.com/stretchr/testify/assert"
+ "github.com/stretchr/testify/require"
+)
+
+func TestYAMLBindingBindBody(t *testing.T) {
+ var s struct {
+ Foo string `yaml:"foo"`
+ }
+ err := yamlBinding{}.BindBody([]byte("foo: FOO"), &s)
+ require.NoError(t, err)
+ assert.Equal(t, "FOO", s.Foo)
+}
diff --git a/context.go b/context.go
index d9fcc285..509ce081 100644
--- a/context.go
+++ b/context.go
@@ -393,8 +393,7 @@ func (c *Context) QueryArray(key string) []string {
func (c *Context) getQueryCache() {
if c.queryCache == nil {
- c.queryCache = make(url.Values)
- c.queryCache, _ = url.ParseQuery(c.Request.URL.RawQuery)
+ c.queryCache = c.Request.URL.Query()
}
}
@@ -491,13 +490,8 @@ func (c *Context) PostFormMap(key string) map[string]string {
// 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)
- }
- }
- return c.get(req.PostForm, key)
+ c.getFormCache()
+ return c.get(c.formCache, key)
}
// get is an internal method and returns a map which satisfy conditions.
diff --git a/context_test.go b/context_test.go
index 439e8ee6..f7bb0f51 100644
--- a/context_test.go
+++ b/context_test.go
@@ -676,7 +676,7 @@ func TestContextRenderJSONP(t *testing.T) {
c.JSONP(http.StatusCreated, H{"foo": "bar"})
assert.Equal(t, http.StatusCreated, w.Code)
- assert.Equal(t, "x({\"foo\":\"bar\"})", w.Body.String())
+ assert.Equal(t, "x({\"foo\":\"bar\"});", w.Body.String())
assert.Equal(t, "application/javascript; charset=utf-8", w.Header().Get("Content-Type"))
}
diff --git a/gin.go b/gin.go
index cbdd080e..894cf094 100644
--- a/gin.go
+++ b/gin.go
@@ -338,6 +338,15 @@ func (engine *Engine) RunFd(fd int) (err error) {
return
}
defer listener.Close()
+ err = engine.RunListener(listener)
+ return
+}
+
+// RunListener attaches the router to a http.Server and starts listening and serving HTTP requests
+// through the specified net.Listener
+func (engine *Engine) RunListener(listener net.Listener) (err error) {
+ debugPrint("Listening and serving HTTP on listener what's bind with address@%s", listener.Addr())
+ defer func() { debugPrintError(err) }()
err = http.Serve(listener, engine)
return
}
diff --git a/gin_integration_test.go b/gin_integration_test.go
index 9beec14d..7e270b91 100644
--- a/gin_integration_test.go
+++ b/gin_integration_test.go
@@ -207,6 +207,42 @@ func TestBadFileDescriptor(t *testing.T) {
assert.Error(t, router.RunFd(0))
}
+func TestListener(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)
+ go func() {
+ router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") })
+ assert.NoError(t, router.RunListener(listener))
+ }()
+ // 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 TestBadListener(t *testing.T) {
+ router := New()
+ addr, err := net.ResolveTCPAddr("tcp", "localhost:10086")
+ assert.NoError(t, err)
+ listener, err := net.ListenTCP("tcp", addr)
+ listener.Close()
+ assert.Error(t, router.RunListener(listener))
+}
+
func TestWithHttptestWithAutoSelectedPort(t *testing.T) {
router := New()
router.GET("/example", func(c *Context) { c.String(http.StatusOK, "it worked") })
diff --git a/go.mod b/go.mod
index dbe8afc5..34151852 100644
--- a/go.mod
+++ b/go.mod
@@ -4,14 +4,15 @@ go 1.12
require (
github.com/gin-contrib/sse v0.1.0
- github.com/golang/protobuf v1.3.1
- github.com/json-iterator/go v1.1.6
- github.com/mattn/go-isatty v0.0.8
- 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/go-playground/locales v0.12.1 // indirect
+ github.com/go-playground/universal-translator v0.16.0 // indirect
+ github.com/golang/protobuf v1.3.2
+ github.com/json-iterator/go v1.1.7
+ github.com/leodido/go-urn v1.1.0 // indirect
+ github.com/mattn/go-isatty v0.0.9
+ github.com/stretchr/testify v1.4.0
github.com/ugorji/go/codec v1.1.7
gopkg.in/go-playground/assert.v1 v1.2.1 // indirect
- gopkg.in/go-playground/validator.v8 v8.18.2
+ gopkg.in/go-playground/validator.v9 v9.29.1
gopkg.in/yaml.v2 v2.2.2
)
diff --git a/go.sum b/go.sum
index c1e9f221..7b4ee320 100644
--- a/go.sum
+++ b/go.sum
@@ -1,33 +1,41 @@
-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/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/gin-contrib/sse v0.1.0 h1:Y/yl/+YNO8GZSjAhjMsSuLt29uWRFHdHYUb5lYOV9qE=
github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI=
-github.com/golang/protobuf v1.3.1 h1:YF8+flBXS5eO826T4nzqPrxfhQThhXl0YzfuUPu4SBg=
-github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
-github.com/json-iterator/go v1.1.6 h1:MrUvLMLTMxbqFJ9kzlvat/rYZqZnW3u4wkLzWTaFwKs=
-github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU=
-github.com/mattn/go-isatty v0.0.8 h1:HLtExJ+uU2HOZ+wI0Tt5DtUDrx8yhUqDcp7fYERX4CE=
-github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s=
-github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd h1:TRLaZ9cD/w8PVh93nsPXa1VrQ6jlwL5oN8l14QlcNfg=
-github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
-github.com/modern-go/reflect2 v1.0.1 h1:9f412s+6RmYXLWZSEzVVgPGK7C2PphHj5RJrvfx9AWI=
-github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
+github.com/go-playground/locales v0.12.1 h1:2FITxuFt/xuCNP1Acdhv62OzaCiviiE4kotfhkmOqEc=
+github.com/go-playground/locales v0.12.1/go.mod h1:IUMDtCfWo/w/mtMfIE/IG2K+Ey3ygWanZIBtBW0W2TM=
+github.com/go-playground/universal-translator v0.16.0 h1:X++omBR/4cE2MNg91AoC3rmGrCjJ8eAeUP/K/EKx4DM=
+github.com/go-playground/universal-translator v0.16.0/go.mod h1:1AnU7NaIRDWWzGEKwgtJRd2xk99HeFyHw3yid4rvQIY=
+github.com/golang/protobuf v1.3.2 h1:6nsPYzhq5kReh6QImI3k5qWzO4PEbvbIW2cwSfR/6xs=
+github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U=
+github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
+github.com/json-iterator/go v1.1.7 h1:KfgG9LzI+pYjr4xvmz/5H4FXjokeP+rlHLhv3iH62Fo=
+github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4=
+github.com/leodido/go-urn v1.1.0 h1:Sm1gr51B1kKyfD2BlRcLSiEkffoG96g6TPv6eRoEiB8=
+github.com/leodido/go-urn v1.1.0/go.mod h1:+cyI34gQWZcE1eQU7NVgKkkzdXDQHr1dBMtdAPozLkw=
+github.com/mattn/go-isatty v0.0.9 h1:d5US/mDsogSGW37IV293h//ZFaeajb69h+EHFsv2xGg=
+github.com/mattn/go-isatty v0.0.9/go.mod h1:YNRxwqDuOph6SZLI9vUUz6OYw3QyUt7WiY2yME+cCiQ=
+github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421 h1:ZqeYNhU3OHLH3mGKHDcjJRFFRrJa6eAM5H+CtDdOsPc=
+github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
+github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742 h1:Esafd1046DLDQ0W1YjYsBW+p8U2u7vzgW2SQVmlNazg=
+github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0=
github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM=
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/stretchr/testify v1.4.0 h1:2E4SXV/wtOkTonXsotYi4li6zVWxYlZuYNCXe9XRJyk=
+github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4=
github.com/ugorji/go v1.1.7 h1:/68gy2h+1mWMrwZFeD1kQialdSzAb432dtpeJ42ovdo=
github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw=
github.com/ugorji/go/codec v1.1.7 h1:2SvQaVZ1ouYrrKKwoSk2pzd4A9evlKJb9oTL+OaLUSs=
github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY=
-golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223 h1:DH4skfRX4EBpamg7iV4ZlCpblAHI6s6TDM39bFZumv8=
-golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
+golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a h1:aYOabOQFp6Vj6W1F80affTUvO9UxmJRx8K0gsfABByQ=
+golang.org/x/sys v0.0.0-20190813064441-fde4db37ae7a/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
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 h1:xoYuJVE7KT85PYWrN730RguIQO0ePzVRfFMXadIrXTM=
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/go-playground/validator.v9 v9.29.1 h1:SvGtYmN60a5CVKTOzMSyfzWDeZRxRuGvRQyEAKbw1xc=
+gopkg.in/go-playground/validator.v9 v9.29.1/go.mod h1:+c9/zcJMFNgbLvly1L1V+PpxWdVbfP1avr/N00E2vyQ=
gopkg.in/yaml.v2 v2.2.2 h1:ZCJp+EgiOT7lHqUV2J862kp8Qj64Jo6az82+3Td9dZw=
gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI=
diff --git a/logger.go b/logger.go
index 5ab4639e..fcf90c25 100644
--- a/logger.go
+++ b/logger.go
@@ -22,18 +22,19 @@ const (
forceColor
)
-var (
- green = string([]byte{27, 91, 57, 55, 59, 52, 50, 109})
- white = string([]byte{27, 91, 57, 48, 59, 52, 55, 109})
- yellow = string([]byte{27, 91, 57, 48, 59, 52, 51, 109})
- red = string([]byte{27, 91, 57, 55, 59, 52, 49, 109})
- blue = string([]byte{27, 91, 57, 55, 59, 52, 52, 109})
- magenta = string([]byte{27, 91, 57, 55, 59, 52, 53, 109})
- cyan = string([]byte{27, 91, 57, 55, 59, 52, 54, 109})
- reset = string([]byte{27, 91, 48, 109})
- consoleColorMode = autoColor
+const (
+ green = "\033[97;42m"
+ white = "\033[90;47m"
+ yellow = "\033[90;43m"
+ red = "\033[97;41m"
+ blue = "\033[97;44m"
+ magenta = "\033[97;45m"
+ cyan = "\033[97;46m"
+ reset = "\033[0m"
)
+var consoleColorMode = autoColor
+
// LoggerConfig defines the config for Logger middleware.
type LoggerConfig struct {
// Optional. Default value is gin.defaultLogFormatter
diff --git a/logger_test.go b/logger_test.go
index 9177e1d9..fc53f356 100644
--- a/logger_test.go
+++ b/logger_test.go
@@ -291,14 +291,14 @@ func TestColorForMethod(t *testing.T) {
return p.MethodColor()
}
- 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, 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, 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, 48, 59, 52, 55, 109}), colorForMethod("OPTIONS"), "options should be white")
- assert.Equal(t, string([]byte{27, 91, 48, 109}), colorForMethod("TRACE"), "trace is not defined and should be the reset color")
+ assert.Equal(t, blue, colorForMethod("GET"), "get should be blue")
+ assert.Equal(t, cyan, colorForMethod("POST"), "post should be cyan")
+ assert.Equal(t, yellow, colorForMethod("PUT"), "put should be yellow")
+ assert.Equal(t, red, colorForMethod("DELETE"), "delete should be red")
+ assert.Equal(t, green, colorForMethod("PATCH"), "patch should be green")
+ assert.Equal(t, magenta, colorForMethod("HEAD"), "head should be magenta")
+ assert.Equal(t, white, colorForMethod("OPTIONS"), "options should be white")
+ assert.Equal(t, reset, colorForMethod("TRACE"), "trace is not defined and should be the reset color")
}
func TestColorForStatus(t *testing.T) {
@@ -309,10 +309,10 @@ func TestColorForStatus(t *testing.T) {
return p.StatusCodeColor()
}
- 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(http.StatusMovedPermanently), "3xx should be white")
- 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, green, colorForStatus(http.StatusOK), "2xx should be green")
+ assert.Equal(t, white, colorForStatus(http.StatusMovedPermanently), "3xx should be white")
+ assert.Equal(t, yellow, colorForStatus(http.StatusNotFound), "4xx should be yellow")
+ assert.Equal(t, red, colorForStatus(2), "other things should be red")
}
func TestResetColor(t *testing.T) {
diff --git a/mode.go b/mode.go
index 8aa84aa8..c3c37fdc 100644
--- a/mode.go
+++ b/mode.go
@@ -71,12 +71,18 @@ func DisableBindValidation() {
binding.Validator = nil
}
-// EnableJsonDecoderUseNumber sets true for binding.EnableDecoderUseNumberto to
+// EnableJsonDecoderUseNumber sets true for binding.EnableDecoderUseNumber to
// call the UseNumber method on the JSON Decoder instance.
func EnableJsonDecoderUseNumber() {
binding.EnableDecoderUseNumber = true
}
+// EnableJsonDisallowUnknownFields sets true for binding.EnableDecoderDisallowUnknownFields to
+// call the DisallowUnknownFields method on the JSON Decoder instance.
+func EnableJsonDecoderDisallowUnknownFields() {
+ binding.EnableDecoderDisallowUnknownFields = true
+}
+
// Mode returns currently gin mode.
func Mode() string {
return modeName
diff --git a/mode_test.go b/mode_test.go
index 3dba5150..1b5fb2ff 100644
--- a/mode_test.go
+++ b/mode_test.go
@@ -40,8 +40,22 @@ func TestSetMode(t *testing.T) {
assert.Panics(t, func() { SetMode("unknown") })
}
+func TestDisableBindValidation(t *testing.T) {
+ v := binding.Validator
+ assert.NotNil(t, binding.Validator)
+ DisableBindValidation()
+ assert.Nil(t, binding.Validator)
+ binding.Validator = v
+}
+
func TestEnableJsonDecoderUseNumber(t *testing.T) {
assert.False(t, binding.EnableDecoderUseNumber)
EnableJsonDecoderUseNumber()
assert.True(t, binding.EnableDecoderUseNumber)
}
+
+func TestEnableJsonDecoderDisallowUnknownFields(t *testing.T) {
+ assert.False(t, binding.EnableDecoderDisallowUnknownFields)
+ EnableJsonDecoderDisallowUnknownFields()
+ assert.True(t, binding.EnableDecoderDisallowUnknownFields)
+}
diff --git a/render/json.go b/render/json.go
index 2b07cba0..70506f78 100644
--- a/render/json.go
+++ b/render/json.go
@@ -138,7 +138,7 @@ func (r JsonpJSON) Render(w http.ResponseWriter) (err error) {
if err != nil {
return err
}
- _, err = w.Write([]byte(")"))
+ _, err = w.Write([]byte(");"))
if err != nil {
return err
}
diff --git a/render/render_test.go b/render/render_test.go
index 4cc71972..95a01b63 100644
--- a/render/render_test.go
+++ b/render/render_test.go
@@ -146,7 +146,7 @@ func TestRenderJsonpJSON(t *testing.T) {
err1 := (JsonpJSON{"x", data}).Render(w1)
assert.NoError(t, err1)
- assert.Equal(t, "x({\"foo\":\"bar\"})", w1.Body.String())
+ assert.Equal(t, "x({\"foo\":\"bar\"});", w1.Body.String())
assert.Equal(t, "application/javascript; charset=utf-8", w1.Header().Get("Content-Type"))
w2 := httptest.NewRecorder()
@@ -158,7 +158,7 @@ func TestRenderJsonpJSON(t *testing.T) {
err2 := (JsonpJSON{"x", datas}).Render(w2)
assert.NoError(t, err2)
- assert.Equal(t, "x([{\"foo\":\"bar\"},{\"bar\":\"foo\"}])", w2.Body.String())
+ assert.Equal(t, "x([{\"foo\":\"bar\"},{\"bar\":\"foo\"}]);", w2.Body.String())
assert.Equal(t, "application/javascript; charset=utf-8", w2.Header().Get("Content-Type"))
}
@@ -347,7 +347,17 @@ func TestRenderRedirect(t *testing.T) {
}
w = httptest.NewRecorder()
- assert.Panics(t, func() { assert.NoError(t, data2.Render(w)) })
+ assert.PanicsWithValue(t, "Cannot redirect with status code 200", func() { data2.Render(w) })
+
+ data3 := Redirect{
+ Code: http.StatusCreated,
+ Request: req,
+ Location: "/new/location",
+ }
+
+ w = httptest.NewRecorder()
+ err = data3.Render(w)
+ assert.NoError(t, err)
// only improve coverage
data2.WriteContentType(w)
diff --git a/response_writer_test.go b/response_writer_test.go
index a5e111e5..1f113e74 100644
--- a/response_writer_test.go
+++ b/response_writer_test.go
@@ -29,38 +29,38 @@ func init() {
}
func TestResponseWriterReset(t *testing.T) {
- testWritter := httptest.NewRecorder()
+ testWriter := httptest.NewRecorder()
writer := &responseWriter{}
var w ResponseWriter = writer
- writer.reset(testWritter)
+ writer.reset(testWriter)
assert.Equal(t, -1, writer.size)
assert.Equal(t, http.StatusOK, writer.status)
- assert.Equal(t, testWritter, writer.ResponseWriter)
+ assert.Equal(t, testWriter, writer.ResponseWriter)
assert.Equal(t, -1, w.Size())
assert.Equal(t, http.StatusOK, w.Status())
assert.False(t, w.Written())
}
func TestResponseWriterWriteHeader(t *testing.T) {
- testWritter := httptest.NewRecorder()
+ testWriter := httptest.NewRecorder()
writer := &responseWriter{}
- writer.reset(testWritter)
+ writer.reset(testWriter)
w := ResponseWriter(writer)
w.WriteHeader(http.StatusMultipleChoices)
assert.False(t, w.Written())
assert.Equal(t, http.StatusMultipleChoices, w.Status())
- assert.NotEqual(t, http.StatusMultipleChoices, testWritter.Code)
+ assert.NotEqual(t, http.StatusMultipleChoices, testWriter.Code)
w.WriteHeader(-1)
assert.Equal(t, http.StatusMultipleChoices, w.Status())
}
func TestResponseWriterWriteHeadersNow(t *testing.T) {
- testWritter := httptest.NewRecorder()
+ testWriter := httptest.NewRecorder()
writer := &responseWriter{}
- writer.reset(testWritter)
+ writer.reset(testWriter)
w := ResponseWriter(writer)
w.WriteHeader(http.StatusMultipleChoices)
@@ -68,7 +68,7 @@ func TestResponseWriterWriteHeadersNow(t *testing.T) {
assert.True(t, w.Written())
assert.Equal(t, 0, w.Size())
- assert.Equal(t, http.StatusMultipleChoices, testWritter.Code)
+ assert.Equal(t, http.StatusMultipleChoices, testWriter.Code)
writer.size = 10
w.WriteHeaderNow()
@@ -76,30 +76,30 @@ func TestResponseWriterWriteHeadersNow(t *testing.T) {
}
func TestResponseWriterWrite(t *testing.T) {
- testWritter := httptest.NewRecorder()
+ testWriter := httptest.NewRecorder()
writer := &responseWriter{}
- writer.reset(testWritter)
+ writer.reset(testWriter)
w := ResponseWriter(writer)
n, err := w.Write([]byte("hola"))
assert.Equal(t, 4, n)
assert.Equal(t, 4, w.Size())
assert.Equal(t, http.StatusOK, w.Status())
- assert.Equal(t, http.StatusOK, testWritter.Code)
- assert.Equal(t, "hola", testWritter.Body.String())
+ assert.Equal(t, http.StatusOK, testWriter.Code)
+ assert.Equal(t, "hola", testWriter.Body.String())
assert.NoError(t, err)
n, err = w.Write([]byte(" adios"))
assert.Equal(t, 6, n)
assert.Equal(t, 10, w.Size())
- assert.Equal(t, "hola adios", testWritter.Body.String())
+ assert.Equal(t, "hola adios", testWriter.Body.String())
assert.NoError(t, err)
}
func TestResponseWriterHijack(t *testing.T) {
- testWritter := httptest.NewRecorder()
+ testWriter := httptest.NewRecorder()
writer := &responseWriter{}
- writer.reset(testWritter)
+ writer.reset(testWriter)
w := ResponseWriter(writer)
assert.Panics(t, func() {
diff --git a/tree.go b/tree.go
index 371d5ad1..41947570 100644
--- a/tree.go
+++ b/tree.go
@@ -65,10 +65,9 @@ func min(a, b int) int {
func countParams(path string) uint8 {
var n uint
for i := 0; i < len(path); i++ {
- if path[i] != ':' && path[i] != '*' {
- continue
+ if path[i] == ':' || path[i] == '*' {
+ n++
}
- n++
}
if n >= 255 {
return 255
diff --git a/vendor/vendor.json b/vendor/vendor.json
index fa8fd13a..70b2d9eb 100644
--- a/vendor/vendor.json
+++ b/vendor/vendor.json
@@ -18,6 +18,28 @@
"version": "v0.1",
"versionExact": "v0.1.0"
},
+ {
+ "checksumSHA1": "b4DmyMT9bicTRVJw1hJXHLhIH+0=",
+ "path": "github.com/go-playground/locales",
+ "revision": "f63010822830b6fe52288ee52d5a1151088ce039",
+ "revisionTime": "2018-03-23T16:04:04Z",
+ "version": "v0.12",
+ "versionExact": "v0.12.1"
+ },
+ {
+ "checksumSHA1": "JgF260rC9YpWyY5WEljjimWLUXs=",
+ "path": "github.com/go-playground/locales/currency",
+ "revision": "630ebbb602847eba93e75ae38bbc7bb7abcf1ff3",
+ "revisionTime": "2019-04-30T15:33:29Z"
+ },
+ {
+ "checksumSHA1": "9pKcUHBaVS+360X6h4IowhmOPjk=",
+ "path": "github.com/go-playground/universal-translator",
+ "revision": "b32fa301c9fe55953584134cb6853a13c87ec0a1",
+ "revisionTime": "2017-02-09T16:11:52Z",
+ "version": "v0.16",
+ "versionExact": "v0.16.0"
+ },
{
"checksumSHA1": "Y2MOwzNZfl4NRNDbLCZa6sgx7O0=",
"path": "github.com/golang/protobuf/proto",
@@ -26,6 +48,14 @@
"version": "v1.3",
"versionExact": "v1.3.0"
},
+ {
+ "checksumSHA1": "zNo6yGy/bCJuzkEcP70oEBtOB2M=",
+ "path": "github.com/leodido/go-urn",
+ "revision": "70078a794e8ea4b497ba7c19a78cd60f90ccf0f4",
+ "revisionTime": "2018-05-24T03:26:21Z",
+ "version": "v1.1",
+ "versionExact": "v1.1.0"
+ },
{
"checksumSHA1": "TB2vxux9xQbvsTHOVt4aRTuvSn4=",
"path": "github.com/json-iterator/go",
@@ -82,6 +112,12 @@
"version": "v1.2",
"versionExact": "v1.2.2"
},
+ {
+ "checksumSHA1": "wnEANt4k5X/KGwoFyfSSnpxULm4=",
+ "path": "github.com/stretchr/testify/require",
+ "revision": "f35b8ab0b5a2cef36673838d662e249dd9c94686",
+ "revisionTime": "2018-05-06T18:05:49Z"
+ },
{
"checksumSHA1": "S4ei9eSqVThDio0Jn2sav6yUbvg=",
"path": "github.com/ugorji/go/codec",
@@ -97,12 +133,12 @@
"revisionTime": "2019-05-02T15:41:39Z"
},
{
- "checksumSHA1": "P/k5ZGf0lEBgpKgkwy++F7K1PSg=",
- "path": "gopkg.in/go-playground/validator.v8",
- "revision": "5f1438d3fca68893a817e4a66806cea46a9e4ebf",
- "revisionTime": "2017-07-30T05:02:35Z",
- "version": "v8.18.2",
- "versionExact": "v8.18.2"
+ "checksumSHA1": "ACzc7AkwLtNgKhqtj8V7SGUJgnw=",
+ "path": "gopkg.in/go-playground/validator.v9",
+ "revision": "46b4b1e301c24cac870ffcb4ba5c8a703d1ef475",
+ "revisionTime": "2019-03-31T13:31:25Z",
+ "version": "v9.28",
+ "versionExact": "v9.28.0"
},
{
"checksumSHA1": "QqDq2x8XOU7IoOR98Cx1eiV5QY8=",