【问题标题】:How do I send a JSON string in a POST request in Go如何在 Go 的 POST 请求中发送 JSON 字符串
【发布时间】:2022-03-27 01:15:58
【问题描述】:

我尝试使用 Apiary 并制作了一个通用模板来将 JSON 发送到模拟服务器并拥有以下代码:

package main

import (
    "encoding/json"
    "fmt"
    "github.com/jmcvetta/napping"
    "log"
    "net/http"
)

func main() {
    url := "http://restapi3.apiary.io/notes"
    fmt.Println("URL:>", url)

    s := napping.Session{}
    h := &http.Header{}
    h.Set("X-Custom-Header", "myvalue")
    s.Header = h

    var jsonStr = []byte(`
{
    "title": "Buy cheese and bread for breakfast."
}`)

    var data map[string]json.RawMessage
    err := json.Unmarshal(jsonStr, &data)
    if err != nil {
        fmt.Println(err)
    }

    resp, err := s.Post(url, &data, nil, nil)
    if err != nil {
        log.Fatal(err)
    }
    fmt.Println("response Status:", resp.Status())
    fmt.Println("response Headers:", resp.HttpResponse().Header)
    fmt.Println("response Body:", resp.RawText())

}

这段代码没有正确发送 JSON,但我不知道为什么。 JSON 字符串在每次调用中都可能不同。我不能为此使用Struct

【问题讨论】:

  • 我不熟悉您使用的一些库,但据我了解,您正在尝试发送 Jsons 的地图。为什么不直接发送带有 json 的字符串?
  • 如果要发送 json,为什么要解组 json?
  • 一个小提示,您可以将消息创建为 struct 或 map[string]interface{} 以添加您想要的所有值,然后使用 json.Marshall 将 map 或 struct 转换为 json .
  • @topo,我研究了小睡的源代码,如果设置了有效负载,他们会调用json.Marshall,我不知道为什么它对他不起作用。

标签: json rest go apiary


【解决方案1】:

我不熟悉打盹,但是使用 Golang 的 net/http 包可以正常工作(playground):

func main() {
    url := "http://restapi3.apiary.io/notes"
    fmt.Println("URL:>", url)

    var jsonStr = []byte(`{"title":"Buy cheese and bread for breakfast."}`)
    req, err := http.NewRequest("POST", url, bytes.NewBuffer(jsonStr))
    req.Header.Set("X-Custom-Header", "myvalue")
    req.Header.Set("Content-Type", "application/json")

    client := &http.Client{}
    resp, err := client.Do(req)
    if err != nil {
        panic(err)
    }
    defer resp.Body.Close()

    fmt.Println("response Status:", resp.Status)
    fmt.Println("response Headers:", resp.Header)
    body, _ := ioutil.ReadAll(resp.Body)
    fmt.Println("response Body:", string(body))
}

【讨论】:

  • 现在它在操场上出现了恐慌。可能你要修复或更新一些东西吗?
  • @Altenrion 不能在操场上工作,我只是用它来粘贴代码,你不能从它打开外部连接。
  • @Altenrion +1 提供可靠的乐队名称建议。
  • 只是一个警告,不要忘记默认情况下golang http客户端永远不会超时,所以对于现实世界,最好按照client.Timeout = time.Second * 15的方式设置一些东西
  • 可以更新它以收集/检查所有错误吗?这(至少对我来说)是谷歌在 Go 中发出帖子请求的最高结果,这是一个很好的答案,但我看到大量忽略错误的示例代码,我认为它鼓励了新手的不良做法。话又说回来,如果有人经常忽略错误,我想他们最终会知道为什么不这样做,但为什么不从一开始就鼓励好的做法呢?
【解决方案2】:

您可以使用post 发布您的 json。

values := map[string]string{"username": username, "password": password}

jsonValue, _ := json.Marshal(values)

resp, err := http.Post(authAuthenticatorUrl, "application/json", bytes.NewBuffer(jsonValue))

【讨论】:

  • 我收到此错误:cannot use jsonValue (type []byte) as type io.Reader in argument to http.Post: []byte does not implement io.Reader (missing Read method)
  • @MandarVaze 我认为io.Readerhttp.Post 可能会出错,而 bytes.NewBuffer() 在我的代码中运行良好
  • 如果重要的话,我正在使用 1.7。 @OneOfOne 列出的代码有效(也使用bytes.NewBuffer(),但使用http.NewRequest 而不是http.Post
  • 根据golang.org/pkg/net/http/#Post,“调用者在读取完resp.Body后应该关闭它。如果提供的正文是io.Closer,它会在请求后关闭。”作为 Go 新手,我如何判断 body 是 io.Closer,或者换句话说,这个例子是否安全?
  • 这种方法的唯一限制是您不能设置自定义标题:/
【解决方案3】:

如果你已经有一个结构体。

import (
    "bytes"
    "encoding/json"
    "io"
    "net/http"
    "os"
)

// .....

type Student struct {
    Name    string `json:"name"`
    Address string `json:"address"`
}

// .....

body := &Student{
    Name:    "abc",
    Address: "xyz",
}

payloadBuf := new(bytes.Buffer)
json.NewEncoder(payloadBuf).Encode(body)
req, _ := http.NewRequest("POST", url, payloadBuf)

client := &http.Client{}
res, e := client.Do(req)
if e != nil {
    return e
}

defer res.Body.Close()

fmt.Println("response Status:", res.Status)
// Print the body to the stdout
io.Copy(os.Stdout, res.Body)

完整的gist

【讨论】:

    【解决方案4】:

    除了标准的net/http包,你可以考虑使用我的GoRequest,它包裹了net/http,让你的生活更轻松,而不用过多考虑json或struct。但是您也可以在一个请求中混合和匹配它们! (您可以在 gorequest github 页面中查看有关它的更多详细信息)

    所以,最后你的代码会变成这样:

    func main() {
        url := "http://restapi3.apiary.io/notes"
        fmt.Println("URL:>", url)
        request := gorequest.New()
        titleList := []string{"title1", "title2", "title3"}
        for _, title := range titleList {
            resp, body, errs := request.Post(url).
                Set("X-Custom-Header", "myvalue").
                Send(`{"title":"` + title + `"}`).
                End()
            if errs != nil {
                fmt.Println(errs)
                os.Exit(1)
            }
            fmt.Println("response Status:", resp.Status)
            fmt.Println("response Headers:", resp.Header)
            fmt.Println("response Body:", body)
        }
    }
    

    这取决于您要如何实现。我制作这个库是因为我和你有同样的问题,我想要更短的代码,易于与 json 一起使用,并且在我的代码库和生产系统中更易于维护。

    【讨论】:

    • 如果 GoRequest 包装了 net/http。是否可以添加它来禁用 TLS 的不安全证书? tr := &http.Transport{ TLSClientConfig: &tls.Config{InsecureSkipVerify: true}, }
    • @user1513388 在任何场景中以任何语言提供跳过 TLS 验证的代码示例总是一个糟糕的主意......你不小心让访问 StackOverflow 的新手长期复制/粘贴了很多“变通办法”并不要'不明白为什么修复 TLS 错误至关重要。修复您的证书导入路径(如果使用自签名进行测试,请导入这些路径)或修复您机器的证书链,或者找出您的服务器显示客户端无法验证的无效证书的原因。
    • 我不太喜欢这个答案的一件事是它构成 JSON 对象的方式,这可能通过注入来利用。更好的方法是组合一个对象,然后将其转换为 JSON(使用适当的转义)。
    • @JohnWhite 我同意,感觉很 ruby​​/js/pythonic
    【解决方案5】:

    io.Pipe 用于another answer 中提到的大型请求正文。这种方法通过将数据从 JSON 编码器流式传输到网络来避免在内存中构建整个请求体。

    这个答案建立在另一个答案的基础上,展示了如何处理错误。始终处理错误!

    • 使用pipe's CloseWithError 函数将编码错误传播回从http.Post 返回的错误。
    • 处理http.Post返回的错误
    • 关闭响应正文。

    代码如下:

    r, w := io.Pipe()
    
    go func() {
        w.CloseWithError(json.NewEncoder(w).Encode(data))
    }()
    
    // Ensure that read side of pipe is closed. This
    // unblocks goroutine in scenario where http.Post
    // errors out before reading the entire request body.
    defer r.Close()
    
    resp, err := http.Post(url, r)
    if err != nil {
        // Adjust error handling here to meet application requrirements.
        log.Fatal(err)
    }
    defer resp.Body.Close()
    // Use the response here.
    

    【讨论】:

      【解决方案6】:

      http 或 https 的示例发布请求

          //Encode the data
             postBody, _ := json.Marshal(map[string]string{
                "name":  "Test",
                "email": "Test@Test.com",
             })
             responseBody := bytes.NewBuffer(postBody)
          //Leverage Go's HTTP Post function to make request
             resp, err := http.Post("https://postman-echo.com/post", "application/json", responseBody)
          //Handle Error
             if err != nil {
                log.Fatalf("An Error Occured %v", err)
             }
             defer resp.Body.Close()
          //Read the response body
             body, err := ioutil.ReadAll(resp.Body)
             if err != nil {
                log.Fatalln(err)
             }
             sb := string(body)
             log.Printf(sb)
      

      【讨论】:

        【解决方案7】:

        如果你有很多数据要发送,你可以使用管道:

        package main
        
        import (
           "encoding/json"
           "io"
           "net/http"
        )
        
        func main() {
           m := map[string]int{"SNG_ID": 75498415}
           r, w := io.Pipe()
           go func() {
              json.NewEncoder(w).Encode(m)
              w.Close()
           }()
           http.Post("https://stackoverflow.com", "application/json", r)
        }
        

        https://golang.org/pkg/io#Pipe

        【讨论】:

          【解决方案8】:

          如果你想这样做,你需要使用这个映射来解组 json 字符串。

          var data map[string]interface{}
          

          但是如果你每次都需要修改json,为了让你的requst body初始化更加方便,你可以使用这个map来创建json body。

          var bodyJsonMap map[string]interface{}{
              "key1": val1,
              "key2": val2,
              ...
          }
          

          然后将其编组为 json 字符串。

          【讨论】:

            猜你喜欢
            • 2016-03-28
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 1970-01-01
            • 2016-07-19
            • 2018-05-27
            相关资源
            最近更新 更多