Подтвердить что ты не робот

Как отправить строку JSON в запрос POST в Go

Я пробовал работать с 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 для этого.

4b9b3361

Ответ 1

Я не знаком с дремотой, но с использованием пакета Golang net/http отлично работает (игровая площадка):

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))
}

Ответ 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))

Ответ 3

Помимо стандартного пакета net/http, вы можете использовать мой GoRequest, который обертывает сеть /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 и более удобен в моей кодовой базе и производственной системе.

Ответ 4

Если у вас уже есть структура.

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

// .....

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

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

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)