Skip to content

Commit 706b2a8

Browse files
committed
Add YAML binding methods to Context.
1 parent fafb393 commit 706b2a8

File tree

3 files changed

+48
-3
lines changed

3 files changed

+48
-3
lines changed

README.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -529,18 +529,18 @@ func main() {
529529

530530
### Model binding and validation
531531

532-
To bind a request body into a type, use model binding. We currently support binding of JSON, XML and standard form values (foo=bar&boo=baz).
532+
To bind a request body into a type, use model binding. We currently support binding of JSON, XML, YAML and standard form values (foo=bar&boo=baz).
533533

534534
Gin uses [**go-playground/validator.v8**](https://github.com/go-playground/validator) for validation. Check the full docs on tags usage [here](http://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Baked_In_Validators_and_Tags).
535535

536536
Note that you need to set the corresponding binding tag on all fields you want to bind. For example, when binding from JSON, set `json:"fieldname"`.
537537

538538
Also, Gin provides two sets of methods for binding:
539539
- **Type** - Must bind
540-
- **Methods** - `Bind`, `BindJSON`, `BindXML`, `BindQuery`
540+
- **Methods** - `Bind`, `BindJSON`, `BindXML`, `BindQuery`, `BindYAML`
541541
- **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.
542542
- **Type** - Should bind
543-
- **Methods** - `ShouldBind`, `ShouldBindJSON`, `ShouldBindXML`, `ShouldBindQuery`
543+
- **Methods** - `ShouldBind`, `ShouldBindJSON`, `ShouldBindXML`, `ShouldBindQuery`, `ShouldBindYAML`
544544
- **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.
545545

546546
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`.

context.go

+11
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ const (
3232
MIMEPOSTForm = binding.MIMEPOSTForm
3333
MIMEMultipartPOSTForm = binding.MIMEMultipartPOSTForm
3434
BodyBytesKey = "_gin-gonic/gin/bodybyteskey"
35+
MIMEYAML = binding.MIMEYAML
3536
)
3637

3738
const abortIndex int8 = math.MaxInt8 / 2
@@ -524,6 +525,11 @@ func (c *Context) BindQuery(obj interface{}) error {
524525
return c.MustBindWith(obj, binding.Query)
525526
}
526527

528+
// BindYAML is a shortcut for c.MustBindWith(obj, binding.YAML).
529+
func (c *Context) BindYAML(obj interface{}) error {
530+
return c.MustBindWith(obj, binding.YAML)
531+
}
532+
527533
// MustBindWith binds the passed struct pointer using the specified binding engine.
528534
// It will abort the request with HTTP 400 if any error ocurrs.
529535
// See the binding package.
@@ -563,6 +569,11 @@ func (c *Context) ShouldBindQuery(obj interface{}) error {
563569
return c.ShouldBindWith(obj, binding.Query)
564570
}
565571

572+
// ShouldBindYAML is a shortcut for c.ShouldBindWith(obj, binding.YAML).
573+
func (c *Context) ShouldBindYAML(obj interface{}) error {
574+
return c.ShouldBindWith(obj, binding.YAML)
575+
}
576+
566577
// ShouldBindWith binds the passed struct pointer using the specified binding engine.
567578
// See the binding package.
568579
func (c *Context) ShouldBindWith(obj interface{}, b binding.Binding) error {

context_test.go

+34
Original file line numberDiff line numberDiff line change
@@ -1380,6 +1380,23 @@ func TestContextBindWithQuery(t *testing.T) {
13801380
assert.Equal(t, 0, w.Body.Len())
13811381
}
13821382

1383+
func TestContextBindWithYAML(t *testing.T) {
1384+
w := httptest.NewRecorder()
1385+
c, _ := CreateTestContext(w)
1386+
1387+
c.Request, _ = http.NewRequest("POST", "/", bytes.NewBufferString("foo: bar\nbar: foo"))
1388+
c.Request.Header.Add("Content-Type", MIMEXML) // set fake content-type
1389+
1390+
var obj struct {
1391+
Foo string `yaml:"foo"`
1392+
Bar string `yaml:"bar"`
1393+
}
1394+
assert.NoError(t, c.BindYAML(&obj))
1395+
assert.Equal(t, "foo", obj.Bar)
1396+
assert.Equal(t, "bar", obj.Foo)
1397+
assert.Equal(t, 0, w.Body.Len())
1398+
}
1399+
13831400
func TestContextBadAutoBind(t *testing.T) {
13841401
w := httptest.NewRecorder()
13851402
c, _ := CreateTestContext(w)
@@ -1470,6 +1487,23 @@ func TestContextShouldBindWithQuery(t *testing.T) {
14701487
assert.Equal(t, 0, w.Body.Len())
14711488
}
14721489

1490+
func TestContextShouldBindWithYAML(t *testing.T) {
1491+
w := httptest.NewRecorder()
1492+
c, _ := CreateTestContext(w)
1493+
1494+
c.Request, _ = http.NewRequest("POST", "/", bytes.NewBufferString("foo: bar\nbar: foo"))
1495+
c.Request.Header.Add("Content-Type", MIMEXML) // set fake content-type
1496+
1497+
var obj struct {
1498+
Foo string `yaml:"foo"`
1499+
Bar string `yaml:"bar"`
1500+
}
1501+
assert.NoError(t, c.ShouldBindYAML(&obj))
1502+
assert.Equal(t, "foo", obj.Bar)
1503+
assert.Equal(t, "bar", obj.Foo)
1504+
assert.Equal(t, 0, w.Body.Len())
1505+
}
1506+
14731507
func TestContextBadAutoShouldBind(t *testing.T) {
14741508
w := httptest.NewRecorder()
14751509
c, _ := CreateTestContext(w)

0 commit comments

Comments
 (0)