Как обслуживать ответ JSON с помощью Go?
Вопрос: В настоящее время я распечатываю свой ответ в 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, "", " ")
}
Ответы
Ответ 1
Вы можете настроить заголовок своего содержимого, чтобы клиенты знали, что json
w.Header().Set("Content-Type", "application/json")
Другой способ маршалирования структуры json - построить кодировщик с помощью http.ResponseWriter
// get a payload p := Payload{d}
json.NewEncoder(w).Encode(p)
Ответ 2
Вы можете сделать что-то подобное в вашей функции getJsonResponse
-
jData, err := json.Marshal(Data)
if err != nil {
panic(err)
return
}
w.Header().Set("Content-Type", "application/json")
w.Write(jData)
Ответ 3
Другие пользователи отмечают, что при кодировании 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)
}
Ответ 4
В gobuffalo.io framework я получил его для работы следующим образом:
// 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.
Ответ 5
Вы можете использовать этот средство отображения пакетов, я написал для решения этой проблемы, это оболочка для обслуживания JSON, JSONP, XML, HTML и т.д.