为了账号安全,请及时绑定邮箱和手机立即绑定

Golang vs PHP https 调用,标头结果不同

Golang vs PHP https 调用,标头结果不同

Go
泛舟湖上清波郎朗 2021-06-29 13:23:23
我正在尝试在 Google App Engine Go 中实现Vault of Satoshi 的 API。他们的参考 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)))))}这两段代码为相同的输入生成相同的签名。但是,当我运行 PHP 代码(使用正确的密钥和秘密)时,服务器以正确的响应进行响应,但是当我运行 Go 代码时,服务器以“无效签名”响应。这个错误表明 Go 生成的 HTTP 请求一定是格式错误的 - HTTP Header 的值是错误的(如果头值完全丢失,则会出现不同的错误),或者 POST 字段的编码方式由于某种原因是错误的。谁能帮我找出为什么这两段代码生成不同的 HTTP 请求的原因,以及如何让 Go 生成像 PHP 代码一样的请求?
查看完整描述

1 回答

?
当年话下

TA贡献1890条经验 获得超9个赞

请参阅 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.Values是map[string][]string。因此,您应该使用 Encode() 一次并在正文和签名中使用相同的结果。有可能通过使用 Encode() 两次,顺序可能会有所不同。这是 Go 和 PHP 之间的一个重要区别。


你也应该养成处理的习惯,error而不是忽视它。


查看完整回答
反对 回复 2021-07-12
  • 1 回答
  • 0 关注
  • 209 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信