Ruby 与 Go/sha256 hmac base64 编码字符串不匹配

标签 ruby go openssl base64 hmac

玩弄虚构,我正在尝试创建一个 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/

相关文章:

go - 管理通过非标准端口访问的自定义 Go 模块

java - 如何创建用于 Apache Tomcat 的 X509 自签名证书

javascript - Ruby on Rails 如何从 CoffeeScript 中的 Controller 注入(inject)值

ruby - 查找数组的相邻元素之间的差异

ruby-on-rails - 使用 rails/spring 重置类变量

ruby-on-rails - 递归修改嵌套哈希中的值

unit-testing - 如何测试Go中的数据库错误?

go - "import cycle not allowed"去安装

c - openssl 1.1 中是否弃用了 BIGNUM BN_ 函数?

c++ - Openssl需要使用CA捆绑文件(中级证书)