将C#散列函数转换为PHP

时间:2012-01-09 10:13:09

标签: c# php hmac

我在C#中有以下功能,这是一个我需要转换为PHP的散列函数。我在PHP中尝试过一些东西,但是我没有得到相同的结果(我对.NET一点也不好)

private static string GetSignature(string args, string privatekey)
{
    var encoding = new System.Text.ASCIIEncoding();
    byte[] key = encoding.GetBytes(privatekey);
    var myhmacsha256 = new HMACSHA256(key);
    byte[] hashValue = myhmacsha256.ComputeHash(encoding.GetBytes(args));
    string hmac64 = Convert.ToBase64String(hashValue);
    myhmacsha256.Clear();
    return hmac64;
} 

PHP中的一次(错误)尝试是:

function encode($data,$key)
{
    return base64_encode( hash_hmac('sha256', $data, $key ) );
}

答案

DampeS8N下面建议的略有变化对我有用。

function encode($data,$key)
{
    iconv_set_encoding("input_encoding", "ASCII");
    iconv_set_encoding("internal_encoding", "ASCII");
    iconv_set_encoding("output_encoding", "ASCII");

    return base64_encode( hash_hmac('sha256', $data, $key, true ) );
}

请不要使用hash_hmac的第四个参数 - 现在将原始输出设置为true作为二进制数据

1 个答案:

答案 0 :(得分:6)

我怀疑.net代码的第一行是罪魁祸首。 PHP has no encoding for the string本身,所以当它需要哈希时,它是散列内部PHP字符串格式的字节(不太可能,其他人可以确认?)或者更可能是先转换为其他东西。在这种情况下,可能是unicode,它与字符串在.net中请求的字符串完全不同。

我的建议是确保PHP也使用ASCII with iconv来实现互操作性。

function encode($data,$key)
{
    return base64_encode( hash_hmac('sha256', iconv( iconv_get_encoding( "internal_encoding"), "ASCII", $data ), iconv( iconv_get_encoding( "internal_encoding"), "ASCII", $key ) ) );
}

我不能确定上面的代码会输出所需的哈希值,但是,因为我没有.net方便测试初始代码。但这可能会指出你正确的方向。

如果这不起作用,iconv_get_encoding(内的值可能是罪魁祸首,请尝试"output_encoding""input_encoding"。您也可能需要使用iconv_set_encoding(将这些相同的值设置为ASCII。

祝你好运!


更新!这是最终奏效的:

function encode($data,$key)
{
    iconv_set_encoding("input_encoding", "ASCII");
    iconv_set_encoding("internal_encoding", "ASCII");
    iconv_set_encoding("output_encoding", "ASCII");

    return base64_encode( hash_hmac('sha256', $data, $key, true ) );
}