Как отправить ответ JSON с помощью Go?

97

Вопрос: В настоящее время я распечатываю свой ответ func Index таким образом, fmt.Fprintf(w, string(response)) однако, как я могу правильно отправить JSON в запросе, чтобы он мог быть использован представлением?

package main

import (
    "fmt"
    "github.com/julienschmidt/httprouter"
    "net/http"
    "log"
    "encoding/json"
)

type Payload struct {
    Stuff Data
}
type Data struct {
    Fruit Fruits
    Veggies Vegetables
}
type Fruits map[string]int
type Vegetables map[string]int


func Index(w http.ResponseWriter, r *http.Request, _ httprouter.Params) {
    response, err := getJsonResponse();
    if err != nil {
        panic(err)
    }
    fmt.Fprintf(w, string(response))
}


func main() {
    router := httprouter.New()
    router.GET("/", Index)
    log.Fatal(http.ListenAndServe(":8080", router))
}

func getJsonResponse()([]byte, error) {
    fruits := make(map[string]int)
    fruits["Apples"] = 25
    fruits["Oranges"] = 10

    vegetables := make(map[string]int)
    vegetables["Carrats"] = 10
    vegetables["Beets"] = 0

    d := Data{fruits, vegetables}
    p := Payload{d}

    return json.MarshalIndent(p, "", "  ")
}
Мэтью Харвуд
источник
github.com/unrolled/render также может помочь.
elithrar

Ответы:

130

Вы можете установить заголовок типа содержимого, чтобы клиенты знали, что ожидать json

w.Header().Set("Content-Type", "application/json")

Другой способ маршалировать структуру в json - создать кодировщик с использованием http.ResponseWriter

// get a payload p := Payload{d}
json.NewEncoder(w).Encode(p)
dm03514
источник
11
Хотя w.Header().Set("Content-Type", "application/json")это правильно для установки типа контента, это не так, когда json.NewEncoderвместо этого я получаю результат txt / plain. Кто-нибудь еще это понимает. Ответ от @poorva сработал, как ожидалось
Jaybeecave
2
Сотрите это. Если я использую, w.WriteHeader(http.StatusOk) я получаю вышеуказанный результат.
Jaybeecave
4
Если я использую, w.WriteHeader(http.StatusOk)то получаю text/plain; charset=utf-8, если я не устанавливаю код состояния явно, я получаю, applicaton/jsonа в ответе все еще есть код состояния 200.
Рамон Рэмбо
1
Хммм ... это может быть связано с документами здесь ? Changing the header map after a call to WriteHeader (or Write) has no effect unless the modified headers are trailers.
Дэн Эспарза
2
Добавление w.Header().Set("Content-Type", "application/json")вышеупомянутой json.NewEncoder(w).Encode(p)работы для меня
Арди Нусаван
35

Другие пользователи комментируют, что Content-Typeэто plain/textпри кодировании. Вы должны установить Content-Typeсначала w.Header().Set, затем код ответа HTTP w.WriteHeader.

Если w.WriteHeaderсначала позвоните, то позвоните w.Header().Setпосле того, как получите plain/text.

Пример обработчика может выглядеть так;

func SomeHandler(w http.ResponseWriter, r *http.Request) {
    data := SomeStruct{}
    w.Header().Set("Content-Type", "application/json")
    w.WriteHeader(http.StatusCreated)
    json.NewEncoder(w).Encode(data)
}
Дэниел Р.
источник
Как вернуть ответ, если моя программа паникует? Я попытался использовать recovery (), а затем вернулся из них, но это не сработало.
infiniteLearner
28

Вы можете сделать что-то подобное в своей getJsonResponseфункции -

jData, err := json.Marshal(Data)
if err != nil {
    // handle error
}
w.Header().Set("Content-Type", "application/json")
w.Write(jData)
бедва
источник
2
Одно важное замечание об этой версии заключается в том, что она использует байтовый фрагмент jData, возможно, без необходимости. Dataможет иметь произвольный размер, в зависимости от маршалируемых данных, поэтому это может быть нетривиальной потерей памяти. После маршаллинга мы копируем из памяти в ResponseWriterпоток. Ответ, использующий json.NewEncoder () и т. Д., Запишет упорядоченный JSON прямо в ResponseWriter(в свой поток ..)
Джонно
1
Сработало у меня! Столкнулся с проблемой, когда w.WriteHeader (http.StatusCreated) был добавлен до или после.
darkdefender 27
1
Нет необходимости возвращаться после паники, так как это выходит из вашей программы,
Андерсфиллинг,
По крайней мере, это решение не добавляет завершающий \ n Encoder.Encode()функции
Джонатан Мюллер
2

В рамках gobuffalo.io я заставил его работать так:

// say we are in some resource Show action
// some code is omitted
user := &models.User{}
if c.Request().Header.Get("Content-type") == "application/json" {
    return c.Render(200, r.JSON(user))
} else {
    // Make user available inside the html template
    c.Set("user", user)
    return c.Render(200, r.HTML("users/show.html"))
}

а затем, когда я хочу получить ответ JSON для этого ресурса, я должен установить «Content-type» на «application / json», и он работает.

Я думаю, что в Rails есть более удобный способ обработки нескольких типов ответов, я пока не видел такого в gobuffalo.

Алекс Ткаченко
источник
0

Вы можете использовать этот рендерер пакетов , я написал для решения такого рода проблем, это оболочка для обслуживания JSON, JSONP, XML, HTML и т. Д.


источник