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

Как обслуживать ответ 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, "", "  ")
}
4b9b3361

Ответ 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 и т.д.