如何为SHA256获取Ruby生成的HMAC,以便安全地匹配Java?

时间:2012-03-16 20:55:03

标签: java ruby sha256 hmac

我有一个运行一些Java代码的tomcat服务器,允许用户使用API​​密钥进行身份验证。该请求使用使用SHA256创建的HMAC。我有一个Ruby客户端,我用它来发出请求,因为我刚接触加密,我很难让它生成匹配的HMAC。我试过不把它设为URL安全,并且匹配。所以我真的很想知道如何让Ruby客户端与URL安全版本匹配(因为我无法更改Java代码)。最后只有一个额外的=字符。提前感谢您的帮助。

对于Ruby我使用1.9.3而对于Java我使用的是6u31以及来自apache的commons-codec-1.6.jar库。

代码

红宝石:

require "openssl"
require "base64"

json_str = "{'community':'LG7B734A', 'login_id':'user1', 'time':'1331928899'}"
digest = OpenSSL::Digest::Digest.new("sha256")
key = [ "4cc45e4258121c3fec84147673e1bd88e51b1c177aafcfa2da72bd4655c9f933" ]
hmac = OpenSSL::HMAC.digest(digest, key.pack("H*"), json_str)

encoded_url_safe = Base64.urlsafe_encode64(hmac)
encoded = Base64.encode64(hmac)

puts("Encoded (Url Safe): " + encoded_url_safe)
puts("Encoded           : " + encoded)

爪哇:

import org.apache.commons.codec.binary.Base64;
import org.apache.commons.codec.binary.Hex;

import javax.crypto.spec.SecretKeySpec;
import javax.crypto.Mac;

public class ExampleHMAC
{
    public static void main(String[] args) throws Exception
    {
        String key = "4cc45e4258121c3fec84147673e1bd88e51b1c177aafcfa2da72bd4655c9f933";
        byte[] keyBytes = Hex.decodeHex(key.toCharArray());

        SecretKeySpec keySpec = new SecretKeySpec(keyBytes, "HmacSHA256");
        Mac mac = Mac.getInstance("HmacSHA256");
        mac.init(keySpec);

        String jsonStr = "{'community':'LG7B734A', 'login_id':'user1', 'time':'1331928899'}";
        byte[] hmacBytes = mac.doFinal(jsonStr.getBytes());

        String encodedUrlSafe = Base64.encodeBase64URLSafeString(hmacBytes);
        String encoded = Base64.encodeBase64String(hmacBytes);

        System.out.println("Encoded (Url Safe): " + encodedUrlSafe);
        System.out.println("Encoded           : " + encoded);
    }
}

输出

红宝石:

Encoded (Url Safe): QgYLqGm1M4qozdEjGC_CnJ8CdBm2jQpsU85kSWFcjKM=
Encoded           : QgYLqGm1M4qozdEjGC/CnJ8CdBm2jQpsU85kSWFcjKM=

爪哇:

Encoded (Url Safe): QgYLqGm1M4qozdEjGC_CnJ8CdBm2jQpsU85kSWFcjKM
Encoded           : QgYLqGm1M4qozdEjGC/CnJ8CdBm2jQpsU85kSWFcjKM=

1 个答案:

答案 0 :(得分:6)

Ruby不会删除尾随的'=' - 这不是绝对的要求,因为你可以在RFC 4648中读到它只是声明在某些应用程序中删除它们可能是合乎需要的。但除此之外,保证Ruby的URL安全编码与Java完全相同。

所以你要做的唯一事情就是去掉尾随的'==',你可以使用正则表达式:

encoded_url_safe_.gsub!(/=+$/, "")