如何在golang中接收jsonp请求

时间:2017-07-01 05:24:07

标签: javascript jquery go jsonp

我以jsonp格式使用jquery发送一些数据:

console.log(data.location.lng);
console.log(data.location.lat);
console.log(data.accuracy);
var urlgr = "http://127.0.0.1:9220/geo/rec_geo/";
var pfg = {
    //lo: data.location.lng,
    lo: 1.0,
    //la: data.location.lat,
    la: 2.0,
    //ac: data.accuracy,
    ac: 3,
};
$.ajax({
    type: 'GET',
    url: urlgr,
    crossDomain: true,
    data: JSON.stringify(pfg),
    dataType: 'jsonp',
    contentType: "application/json; charset=utf-8",
    success: function (data) {
        console.log("data rec geo");
        console.log(data);

    }
});

这是我在golang中接收它的代码:

type geoData struct {
    Long float32 `json:"lo"`
    Lat   float32  `json:"la"`
    Accuracy  int `json:"ac"`
}

func recordGeo(w http.ResponseWriter, r *http.Request) {
    s := strings.Split(r.RemoteAddr, ":")
    ip := s[0]
    Info.Println("4")
    var geoRec geoData
    decoder := json.NewDecoder(r.Body)
    err := decoder.Decode(&geoRec)
    fmt.Println("geoRec")
    fmt.Println(geoRec)
    checkErr(err)
}

但我有一个EOF错误:

2017/07/01 07:16:22 http: panic serving 127.0.0.1:50680: EOF

所以我的问题是如何在golang中正确接收jsonp请求并将其解析为结构?

这是我在golang服务器端获得的请求:

GET /geo/rec_geo/?callback=jQuery31108538596418163691_1498913991164&{%22lo%22:1,%22la%22:2,%22ac%22:3}&_=1498913991166 HTTP/1.1
Host: 127.0.0.1:9220
user-agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/56.0.2924.87 Safari/537.36
accept-encoding: gzip, deflate, sdch, br
accept-language: fr-FR,fr;q=0.8,en-US;q=0.6,en;q=0.4
referer: http://localhost:8080/
cookie: csrftoken=Y7WA181u22l9sScw9UbPxM38wCGQzjaLMMysAesJSWZm89Pj6CCdBkEF01ibpUjW
alexatoolbar-alx_ns_ph: AlexaToolbar/alx-4.0.1
connection: keep-alive

编辑删除了: Info.Println专线 但我仍然有同样的错误

感谢

3 个答案:

答案 0 :(得分:1)

JSONP是从服务器接收数据而不是相反。

您可以详细了解JSONP - WikipediaSO PostJSONP examples

实际上它并不特定于Go语言,您可以在任何服务器端编程中实现JSONP处理。

出于您的目的,您似乎需要来自JavaScript / Jquery的JSON Post请求。在Go lang处理程序上做 -

func recordGeo(w http.ResponseWriter, req *http.Request) {
    decoder := json.NewDecoder(req.Body)
    var geoRec geoData
    if err := decoder.Decode(&geoRec); err != nil {
        fmt.Println(err)
    }
    defer req.Body.Close()
    fmt.Println(geoRec)
}

答案 1 :(得分:1)

您的golang代码存在问题,如下所示:

decoder := json.NewDecoder(r.Body)
Info.Println(r.Body)
err := decoder.Decode(&geoRec)

Info.Printlndecoder.Decode都从请求正文中读取io.Reader。执行Info.Println后,读者处于EOF状态。从这一点开始,任何进一步的Read都会导致EOF错误,即您在应用程序中看到的错误。如果删除(注释掉)Info.Println,您将能够将数据解码为结构。

修改
我忘了并且没注意JSONP是GET请求。 JSON数据嵌入在查询中。尝试以下方法来提取JSON数据(请根据您的需要进行调整)。

//snippet inside recordGeo function
//...

callback := ""
geoRec := geoData{}
queries := r.URL.Query()

//See http://api.jquery.com/jQuery.ajax/
//for 'jsonp' data type parameters.
for key, val := range queries {
    if key == "callback" {
        callback = val[0]
    } else if key == "_" {
        //the Timestamp
    } else {
        //JSON data in the key
        dec := json.NewDecoder(bytes.NewBufferString(key))
        err := dec.Decode(&geoRec)
        if err != nil {
            log.Println(err)
        }
        log.Printf("DATA=%#v", geoRec)
    }
}

//...
//do some processing
response := "{YOUR RESPONSE DATA}"
if callback != "" {
    w.Header().Set("Content-Type", "application/javascript")
    response = fmt.Sprintf("%s(%s);", callback, response)
}
w.Write([]byte(response))

如果在客户端,您将数据设置为data: pfg(没有JSON.stringify),那么在服务器端,每个参数都可以通过查询参数访问:

//
lon := queries.Get("lo")
lat := queries.Get("la")
acc := queries.Get("ac")

或者,如果您将AJAX请求数据设置为data: "mydata=" + JSON.stringify(pfg),则可以将数据提取为:

mydata := queries.Get("mydata")
dec := json.NewDecoder(bytes.NewBufferString(mydata))
err := dec.Decode(&geoRec)

处理完毕后,您需要向客户端发送适当的JSONP响应,即callback(DATA);

答案 2 :(得分:1)

我遇到了同样的问题。然后,我找到了可以完美处理jsonp请求的echo框架(即github.com/labstack/echo包及其中间件)。这是演示cook for jsonp

希望这会有所帮助。