GoLang WebServer在Json Response上发送param结构的描述

时间:2017-12-29 20:42:02

标签: php json curl go

所以这就是交易:我已经在一个庞大的系统(PHP)上工作了几年,现在,我决定放弃部分繁重的golang脚本工作。

到目前为止,我将一些PHP脚本复制到go版本。然后,我能够对哪个选项更好进行基准测试(好吧,我知道go更快,但我需要curl或socket来实现通信,所以,我必须检查它是否仍然值得。)

其中一个脚本只生成一个随机代码,检查这个新代码是否已经在使用(在mysql db上),如果没有,记录新代码并返回它,如果已经在使用,只需递归调用该函数再次找到一个独家代码。非常简单。

我已经在php中使用了这个代码生成器,因此,在go中使用json params将其称为http / post。 使用linux终端,我称之为

curl -H [headers] -d [jsondata] [host]

我回来了一个非常简单的json

{"locator" : "XXXXXX"}

之后,我编写了一个简单的php脚本来调用脚本并检查每个脚本完成的时间,例如:

<?php
public function indexAction()
    {
    $phpTime = $endPHP = $startPHP =
    $goTime = $endGO = $startGO = 0;

// my standard function already in use


    ob_start();

    $startPHP = microtime(true);
    $MyCodeLocator =  $this->container->get('MyCodeLocator')->Generate(22, 5);
    $endPHP = microtime(true);

    ob_end_clean();


    sleep(1);
    // Lets call using curl
    $data = array("comon" => "22", "lenght" => "5");
    $data_string = json_encode($data);

    ob_start();
    $startGO = microtime(true);



    $ch = curl_init('http://localhost:8888/dev/fibootkt/MyCodeGenerator');
    curl_setopt($ch, CURLOPT_CUSTOMREQUEST, "POST");
    curl_setopt($ch, CURLOPT_POSTFIELDS, $data_string);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
    curl_setopt($ch, CURLOPT_HTTPHEADER, array(
            'Content-Type: application/json',
            'Content-Length: ' . strlen($data_string))
    );

    $result = curl_exec($ch);
    curl_close($ch);
    $endGO = microtime(true);
    ob_end_clean();

    $result_rr = json_decode($result);

    // tst just echo the variable in a nice way, second parameter means no kill execution
    tst($result, 1); // HERE IS MY PROBLEM, please read below
    tst($result_rr, 1); // IT SHOW NULL

    sleep(1);

// just to have params for comparision, always try by shell script
    ob_start();
    $startShell = microtime(true);;

    $exec =  "curl -H \"Content-Type: application/json\" -d '{\"comon\":\"22\"}' http://localhost:8888/dev/fibootkt/MyCodeGenerator";
    $output = shell_exec($exec);
    $endShell  = microtime(true);
    ob_end_clean();

    tst(json_decode($output),1); // here show a stdclass with correct value
    tst($output,1); // here shows a json typed string ready to be converted

    // and here it is just for show the execution time ...
    $phpTime = $endPHP - $startPHP;
    $goTime = $endGO - $startGO ;
    $shellTime = $endShell - $startShell;

    tst("php " . $phpTime, 1);
    tst("curl ". $goTime, 1);
    tst("shell " . $shellTime, 1);

我从GO获得结果: 通过Shell脚本:

{"locator" : "DPEWX22"} 

因此,这个很容易解码为stdobj。

但是,使用curl,操作更快!所以,我想用它。 但是,卷曲请求响应类似于:

{"Value":"string","Type":{},"Offset":26,"Struct":"CodeLocatorParams","Field":"lenght"}
{"locator":"DPEWX22"}

当我尝试解码它时,我得到一个null结果!!!

CodeLocatorParams我使用的struct类型去获取post params,如下所示

所以,这是我的问题:为什么GO会回来这个?如何避免它。

我有另一个类似的go脚本,它不接受params并响应类似的json(但在这种情况下,是一个qrcode图像路径)并且它工作正常!

我的去功能:

type CodeLocatorParams struct {
    Comon string `json:"comon"`
    Lenght int  `json:"lenght"`
}

func Generate(w http.ResponseWriter, r *http.Request) {


    data, err := ioutil.ReadAll(io.LimitReader(r.Body, 1048576))
    if err != nil {
        fmt.Println(err)
        panic(err)

    }
    if err := r.Body.Close(); err != nil {
        panic(err)
    }

// retrieve post data  and set it to params which is CodeLocatorParams type
    var params CodeLocatorParams
    if err := json.Unmarshal(data, &params ); err != nil {
        w.Header().Set("Content-Type", "application/json; charset=UTF-8")
        w.WriteHeader(422) // unprocessable entity
        if err := json.NewEncoder(w).Encode(err); err != nil {
            fmt.Println(err)
            panic(err)
        }
    }





    var result struct{
        Locator string `json:"locator"`
    }
// here actually comes the func that generates random code and return it as string, but for now, just set it static
    result.Locator = "DPEWX22"

    w.Header().Set("Content-Type", "application/json; charset=UTF-8")
    json.NewEncoder(w).Encode(result)


}

1 个答案:

答案 0 :(得分:1)

解析传入的JSON时出错。此错误将作为{"Value":"string","Type":{},"Offset":26,"Struct":"CodeLocatorParams","Field":"lenght"}写入响应。处理程序继续执行并写入正常响应{"locator":"DPEWX22"}

以下是解决问题的方法:

  • 写入错误响应后,从处理程序返回。
  • 输入JSON具有长度作为字符串值。将struct字段从int更改为string或修改输入以传递整数。