玩弄虚构,我正在尝试创建一个 ruby 客户端。
出于安全原因,我需要对 url 进行签名
这是 go 提供的示例:
package main
import (
"crypto/hmac"
"crypto/sha256"
"encoding/base64"
"fmt"
)
func main() {
signKey := "ea79b7fd-287b-4ffe-b941-bf983181783f"
urlPath := "/resize"
url := "https%3A%2F%2Fxyz"
urlQuery := "nocrop=true&type=jpeg&url=" + url + "&width=500"
h := hmac.New(sha256.New, []byte(signKey))
h.Write([]byte(urlPath))
h.Write([]byte(urlQuery))
buf := h.Sum(nil)
fmt.Println(base64.RawURLEncoding.EncodeToString(buf)
}
转换为 ruby,这给了我们:
require 'openssl'
require 'base64'
signKey = "ea79b7fd-287b-4ffe-b941-bf983181783f"
urlPath = "/resize"
url = "https%3A%2F%2Fxyz"
urlQuery = "nocrop=true&type=jpeg&url=" + url + "&width=500"
digest = OpenSSL::Digest.new('sha256')
hmac = OpenSSL::HMAC.digest(digest, signKey, "#{urlPath}#{urlQuery}")
pp Base64.strict_encode64(hmac)
我们快完成了,但是有一个小问题,不知道是因为 openssl 还是 base64,但是例如当我用 go 得到这个时:
wClkWcUvI9ILs7noAr_HtnKpRCeeWBXE1Ne2C99sAco
我得到以下 ruby 版本:
wClkWcUvI9ILs7noAr/HtnKpRCeeWBXE1Ne2C99sAco=
对于 ruby,无论做什么,它都以 =
虽然 go 使用下划线,但 ruby 使用反斜杠(这最后一个陈述可能是对特定 ruby 部分完全不了解的结果,但让我们详细说明这个问题)
应该怎么做才能使两个版本获得相同的输出?为什么我们在这些语言之间得到接近但不准确的结果?
非常感谢回复
最佳答案
Go 代码使用 base64 编码的 URL 安全变体,而您的 Ruby 代码使用普通版本。 URL 安全版本使用 -
和 _
而不是 +
和 /
以便在 URL 中使用是安全的. Ruby 版本还包括填充(末尾的 =
)。
您可以使用 URL safe version in Ruby , 你也可以指定不填充以获得与 Go 相同的结果:
Base64.urlsafe_encode64(hmac, false)
关于Ruby 与 Go/sha256 hmac base64 编码字符串不匹配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56559148/