gpt4 book ai didi

php - Golang vs PHP https 调用具有不同的 header 结果

转载 作者:IT王子 更新时间:2023-10-29 01:34:03 24 4
gpt4 key购买 nike

我正在尝试实现 Vault of Satoshi's API在 Google App Engine Go 中。他们的引用 API 在 PHP 中:

<?php
$serverURL = 'https://api.vaultofsatoshi.com';
$apiKey = 'ENTER_YOUR_API_KEY_HERE';
$apiSecret = 'ENTER_YOUR_API_SECRET_HERE';
function usecTime() {
list($usec, $sec) = explode(' ', microtime());
$usec = substr($usec, 2, 6);
return intval($sec.$usec);
}
$url = 'https://api.vaultofsatoshi.com';
$endpoint = '/info/currency';
$url = $serverURL . $endpoint;

$parameters= array();
$parameters['nonce'] = usecTime();
$data = http_build_query($parameters);

$httpHeaders = array(
'Api-Key: ' . $apiKey,
'Api-Sign:' . base64_encode(hash_hmac('sha512', $endpoint . chr(0) . $data, $apiSecret)),
);
// Initialize the PHP curl agent
$ch = curl_init();
curl_setopt($ch, CURLOPT_USERAGENT, "something specific to me");
curl_setopt($ch, CURLOPT_URL, $url);
curl_setopt($ch, CURLOPT_POST, true);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($ch, CURLOPT_FAILONERROR, true);
curl_setopt($ch, CURLOPT_HTTPHEADER, $httpHeaders);
curl_setopt($ch, CURLOPT_POSTFIELDS, $data);
$output = curl_exec($ch);
curl_close($ch);
echo $output;
?>

我的 Go 代码如下所示:

func GenerateSignatureFromValues(secretKey string, endpoint string, values url.Values) string {
query:=[]byte(values.Encode())
toEncode:=[]byte(endpoint)
toEncode = append(toEncode, 0x00)
toEncode = append(toEncode, query...)
key:=[]byte(secretKey)
hmacHash:=hmac.New(sha512.New, key)
hmacHash.Write(toEncode)
answer := hmacHash.Sum(nil)
return base64.StdEncoding.EncodeToString(([]byte(strings.ToLower(hex.EncodeToString(answer)))))
}

func Call(c appengine.Context) map[string]interface{} {
serverURL:="https://api.vaultofsatoshi.com"
apiKey:="ENTER_YOUR_API_KEY_HERE"
apiSecret:="ENTER_YOUR_API_SECRET_HERE"
endpoint:="/info/order_detail"
tr := urlfetch.Transport{Context: c}
values := url.Values{}
values.Set("nonce", strconv.FormatInt(time.Now().UnixNano()/1000, 10))
signature:=GenerateSignatureFromValues(apiSecret, endpoint, values)
req, _:=http.NewRequest("POST", serverURL+endpoint, nil)
req.Form=values
req.Header.Set("Api-Key", apiKey)
req.Header.Set("Api-Sign", signature)
resp, err:=tr.RoundTrip(req)
if err != nil {
c.Errorf("API post error: %s", err)
return nil
}
defer resp.Body.Close()
body, _:= ioutil.ReadAll(resp.Body)
result := make(map[string]interface{})
json.Unmarshal(body, &result)
return result
}

这两段代码为相同的输入生成相同的签名。但是,当我运行 PHP 代码(使用正确的 key 和 secret )时,服务器会以正确的响应进行响应,但是当我运行 Go 代码时,服务器会以“无效签名”进行响应。此错误表明由 Go 生成的 HTTP 请求一定是格式错误的——HTTP header 的值错误(如果 header 值完全缺失会出现不同的错误),或者 POST 字段的编码方式由于某种原因是错误的。

谁能帮我找出这两段代码生成不同 HTTP 请求的原因,以及如何让 Go 生成像 PHP 代码一样的请求?

最佳答案

请参阅 Request.Form 的文档:

 // Form contains the parsed form data, including both the URL
// field's query parameters and the POST or PUT form data.
// This field is only available after ParseForm is called.
// The HTTP client ignores Form and uses Body instead.
Form url.Values

特别是 “HTTP 客户端忽略 Form 而使用 Body。”

用这一行:

req, _:= http.NewRequest("POST", serverURL+endpoint, nil)

你应该使用它而不是 nil:

bytes.NewBufferString(values.Encode())

另外请记住,map 的顺序是无法保证的。 url.Valuesmap[string][]string。所以你应该使用一次 Encode() 并在正文和签名中使用相同的结果。有可能通过使用 Encode() 两次,顺序可能会有所不同。这是 Go 和 PHP 之间的重要区别。

您还应该养成处理错误而不是忽略它的习惯。

关于php - Golang vs PHP https 调用具有不同的 header 结果,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20616982/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com