Chi空的http.Request.Body in render.Bind

时间:2017-06-20 21:40:59

标签: json http go go-chi

我正在使用github.com/pressly/chi来构建这个简单的程序,我尝试从http.Request.Body解码一些JSON:

package main

import (
    "encoding/json"
    "fmt"
    "net/http"

    "github.com/pressly/chi"
    "github.com/pressly/chi/render"
)

type Test struct {
    Name string `json:"name"`
}

func (p *Test) Bind(r *http.Request) error {
    err := json.NewDecoder(r.Body).Decode(p)
    if err != nil {
        return err
    }
    return nil
}

func main() {
    r := chi.NewRouter()

    r.Post("/products", func(w http.ResponseWriter, r *http.Request) {
        var p Test
        // err := render.Bind(r, &p)
        err := json.NewDecoder(r.Body).Decode(&p)

        if err != nil {
            panic(err)
        }

        fmt.Println(p)
    })

    http.ListenAndServe(":8080", r)
}

当我不使用render.Bind()(来自"github.com/pressly/chi/render")时,它会按预期工作。

但是,当我取消注释行err := render.Bind(r, &p)并对err := json.NewDecoder(r.Body).Decode(&p)行发表评论时,它会对EOF感到恐慌:

  

2017/06/20 22:26:39 http: panic serving 127.0.0.1:39696: EOF

因此json.Decode()失败。

我做错了什么或者http.Request.Body在调用render.Bind()之前是否已经在其他地方读过了?

1 个答案:

答案 0 :(得分:1)

render.Bind的目的是执行解码并执行Bind(r)来执行后解码操作。

例如:

type Test struct {
   Name string `json:"name"`
}

func (p *Test) Bind(r *http.Request) error {
   // At this point, Decode is already done by `chi`
   p.Name = p.Name + " after decode"
  return nil
}

如果您只需要进行JSON解码,则在解码后不需要对解码值进行其他操作。只需使用:

// Use Directly JSON decoder of std pkg
err := json.NewDecoder(r.Body).Decode(&p)

OR

// Use wrapper method from chi DecodeJSON
err := render.DecodeJSON(r.Body, &p)