使用 openssl 的 SHA 512 HMAC 消息身份验证问题
Issues with SHA 512 HMAC message authentication using openssl
我需要验证一个 websocket 端点来订阅私有数据,验证步骤如下:
- 使用 SHA-256 算法散列挑战
- Base64-解码你的api_secret
- 使用步骤 2 的结果用 HMAC-SHA-512 算法对步骤 1 的结果进行散列
- 对第3步的结果进行Base64编码
我在我的 C++ 程序中使用 openssl 进行所有加密,我正在使用我在 Whosebug 上找到的一些 base64 编码和解码算法,但是我无法遵循身份验证程序并产生正确的结果。
我相信 base64 解码器是正确的,因为当我解码秘密时它会生成正确的二进制文件此外,openssl sha256 算法也是正确的,因为它会为挑战生成正确的哈希值,(我使用 cryptiis 在线 base64 解码器和 sha256 来验证这一点),我在最后一步使用 openssl HMAC 或 base64 编码器的方式一定有问题。
#include <openssl/sha.h>
#include <cstdio>
#include <cstring>
void sha256(const char *string, char outputBuffer[65])
{
unsigned char hash[SHA256_DIGEST_LENGTH];
SHA256_CTX sha256;
SHA256_Init(&sha256);
SHA256_Update(&sha256, string, strlen(string));
SHA256_Final(hash, &sha256);
for (int i = 0; i < SHA256_DIGEST_LENGTH; ++i) {
sprintf(outputBuffer + (i * 2), "%02x", hash[i]);
}
outputBuffer[64] = 0;
}
#include <openssl/bio.h>
#include <openssl/evp.h>
#include <cstring>
#include <memory>
#include <string>
#include <vector>
namespace {
struct BIOFreeAll { void operator()(BIO* p) { BIO_free_all(p); } };
}
auto Base64Encode(const std::vector<unsigned char>& binary)
{
std::unique_ptr<BIO,BIOFreeAll> b64(BIO_new(BIO_f_base64()));
BIO_set_flags(b64.get(), BIO_FLAGS_BASE64_NO_NL);
BIO* sink = BIO_new(BIO_s_mem());
BIO_push(b64.get(), sink);
BIO_write(b64.get(), binary.data(), binary.size());
BIO_flush(b64.get());
const unsigned char* encoded;
const unsigned long len = BIO_get_mem_data(sink, &encoded);
return std::basic_string<unsigned char>{encoded, len};
}
// Assumes no newlines or extra characters in encoded string
std::vector<unsigned char> Base64Decode(const char* encoded)
{
std::unique_ptr<BIO,BIOFreeAll> b64(BIO_new(BIO_f_base64()));
BIO_set_flags(b64.get(), BIO_FLAGS_BASE64_NO_NL);
BIO* source = BIO_new_mem_buf(encoded, -1); // read-only source
BIO_push(b64.get(), source);
const int maxlen = strlen(encoded) / 4 * 3 + 1;
std::vector<unsigned char> decoded(maxlen);
const int len = BIO_read(b64.get(), decoded.data(), maxlen);
decoded.resize(len);
return decoded;
}
#include <openssl/hmac.h>
int main(int argc, const char * argv[])
{
const char* challenge = "c100b894-1729-464d-ace1-52dbce11db42";
static char buffer[65];
sha256(challenge, buffer);
printf("%s\n", buffer);
const char* encoded = "7zxMEF5p/Z8l2p2U7Ghv6x14Af+Fx+92tPgUdVQ748FOIrEoT9bgT+bTRfXc5pz8na+hL/QdrCVG7bh9KpT0eMTm";
std::cout << "encoded = " << encoded << std::endl;
const std::vector<unsigned char> decoded = Base64Decode(encoded);
std::cout << "decoded = " << decoded.data() << '\n';
// The data that we're going to hash using HMAC
std::basic_string<unsigned char> data = {decoded.data(), decoded.size()};
unsigned char* digest;
// Using sha512 hash engine here.
// You may use other hash engines. e.g EVP_md5(), EVP_sha224, EVP_sha512, etc
digest = HMAC(EVP_sha512(), data.c_str(), data.size(), reinterpret_cast<unsigned char*>(buffer), strlen(buffer), NULL, NULL);
// Be careful of the length of string with the choosen hash engine. SHA1 produces a 20-byte hash value which rendered as 40 characters.
// Change the length accordingly with your choosen hash engine
char mdString[128];
for(int i = 0; i < 64; ++i)
sprintf(&mdString[i*2], "%02x", (unsigned int)digest[i]);
printf("HMAC digest: %s\n", mdString);
const std::vector<unsigned char> binary{&digest[0], &digest[127] + 1};
const std::basic_string<unsigned char> encoded_result = Base64Encode(binary);
for (unsigned i = 0; i < 64; ++i)
{
std::cout << std::hex << std::setw(2) << (unsigned int)encoded_result[i];
}
std::cout << '\n';
return 0;
}
代码可能无法第一次编译,因为我已经从一个更大的存储库中提取了代码片段,但是如果将所有代码都放入一个文件中,它应该可以编译(或者需要稍作努力才能成功编译)。
当初始挑战的值为
"c100b894-1729-464d-ace1-52dbce11db42"
而 api 秘密是
"7zxMEF5p/Z8l2p2U7Ghv6x14Af+Fx+92tPgUdVQ748FOIrEoT9bgT+bTRfXc5pz8na+hL/QdrCVG7bh9KpT0eMTm"
HMAC 摘要后面的行应该是带符号的输出,我希望它是
"4JEpF3ix66GA2B+ooK128Ift4XQVtc137N9yeg4Kqsn9PI0Kpzbysl9M1IeCEdjg0zl00wkVqcsnG4bm
nlMb3A=="
而实际上是
"336e394b567a55634d46376478344b594354767267636d39456f584f51326c376f334f2f3348796f6939647a7a516a456e41786c3551537541453930422f424b".
更麻烦的是,我可以使用 python 和 C# 复制正确的结果,非常简单地使用库函数,我很不确定我哪里出错了。
您似乎过分喜欢对数据进行十六进制编码!
首先,在您的 sha256
函数中,您正确地散列数据以获得 32 字节的摘要,但是随后您对其进行十六进制编码以获得您稍后使用的 64 个十六进制字符(加上空终止符)作为 HMAC 的输入。您需要使用那些原始的 32 个字节。
然后,在计算 HMAC 并对结果进行 base 64 编码之后,在打印出来之前对 that 进行十六进制编码。没有必要这样做,base 64 已经包含可打印字符。
去掉你进行十六进制编码的那两个循环(并改变 sha256
所以你 return 正确的缓冲区)它应该可以正常工作。
我需要验证一个 websocket 端点来订阅私有数据,验证步骤如下:
- 使用 SHA-256 算法散列挑战
- Base64-解码你的api_secret
- 使用步骤 2 的结果用 HMAC-SHA-512 算法对步骤 1 的结果进行散列
- 对第3步的结果进行Base64编码
我在我的 C++ 程序中使用 openssl 进行所有加密,我正在使用我在 Whosebug 上找到的一些 base64 编码和解码算法,但是我无法遵循身份验证程序并产生正确的结果。
我相信 base64 解码器是正确的,因为当我解码秘密时它会生成正确的二进制文件此外,openssl sha256 算法也是正确的,因为它会为挑战生成正确的哈希值,(我使用 cryptiis 在线 base64 解码器和 sha256 来验证这一点),我在最后一步使用 openssl HMAC 或 base64 编码器的方式一定有问题。
#include <openssl/sha.h>
#include <cstdio>
#include <cstring>
void sha256(const char *string, char outputBuffer[65])
{
unsigned char hash[SHA256_DIGEST_LENGTH];
SHA256_CTX sha256;
SHA256_Init(&sha256);
SHA256_Update(&sha256, string, strlen(string));
SHA256_Final(hash, &sha256);
for (int i = 0; i < SHA256_DIGEST_LENGTH; ++i) {
sprintf(outputBuffer + (i * 2), "%02x", hash[i]);
}
outputBuffer[64] = 0;
}
#include <openssl/bio.h>
#include <openssl/evp.h>
#include <cstring>
#include <memory>
#include <string>
#include <vector>
namespace {
struct BIOFreeAll { void operator()(BIO* p) { BIO_free_all(p); } };
}
auto Base64Encode(const std::vector<unsigned char>& binary)
{
std::unique_ptr<BIO,BIOFreeAll> b64(BIO_new(BIO_f_base64()));
BIO_set_flags(b64.get(), BIO_FLAGS_BASE64_NO_NL);
BIO* sink = BIO_new(BIO_s_mem());
BIO_push(b64.get(), sink);
BIO_write(b64.get(), binary.data(), binary.size());
BIO_flush(b64.get());
const unsigned char* encoded;
const unsigned long len = BIO_get_mem_data(sink, &encoded);
return std::basic_string<unsigned char>{encoded, len};
}
// Assumes no newlines or extra characters in encoded string
std::vector<unsigned char> Base64Decode(const char* encoded)
{
std::unique_ptr<BIO,BIOFreeAll> b64(BIO_new(BIO_f_base64()));
BIO_set_flags(b64.get(), BIO_FLAGS_BASE64_NO_NL);
BIO* source = BIO_new_mem_buf(encoded, -1); // read-only source
BIO_push(b64.get(), source);
const int maxlen = strlen(encoded) / 4 * 3 + 1;
std::vector<unsigned char> decoded(maxlen);
const int len = BIO_read(b64.get(), decoded.data(), maxlen);
decoded.resize(len);
return decoded;
}
#include <openssl/hmac.h>
int main(int argc, const char * argv[])
{
const char* challenge = "c100b894-1729-464d-ace1-52dbce11db42";
static char buffer[65];
sha256(challenge, buffer);
printf("%s\n", buffer);
const char* encoded = "7zxMEF5p/Z8l2p2U7Ghv6x14Af+Fx+92tPgUdVQ748FOIrEoT9bgT+bTRfXc5pz8na+hL/QdrCVG7bh9KpT0eMTm";
std::cout << "encoded = " << encoded << std::endl;
const std::vector<unsigned char> decoded = Base64Decode(encoded);
std::cout << "decoded = " << decoded.data() << '\n';
// The data that we're going to hash using HMAC
std::basic_string<unsigned char> data = {decoded.data(), decoded.size()};
unsigned char* digest;
// Using sha512 hash engine here.
// You may use other hash engines. e.g EVP_md5(), EVP_sha224, EVP_sha512, etc
digest = HMAC(EVP_sha512(), data.c_str(), data.size(), reinterpret_cast<unsigned char*>(buffer), strlen(buffer), NULL, NULL);
// Be careful of the length of string with the choosen hash engine. SHA1 produces a 20-byte hash value which rendered as 40 characters.
// Change the length accordingly with your choosen hash engine
char mdString[128];
for(int i = 0; i < 64; ++i)
sprintf(&mdString[i*2], "%02x", (unsigned int)digest[i]);
printf("HMAC digest: %s\n", mdString);
const std::vector<unsigned char> binary{&digest[0], &digest[127] + 1};
const std::basic_string<unsigned char> encoded_result = Base64Encode(binary);
for (unsigned i = 0; i < 64; ++i)
{
std::cout << std::hex << std::setw(2) << (unsigned int)encoded_result[i];
}
std::cout << '\n';
return 0;
}
代码可能无法第一次编译,因为我已经从一个更大的存储库中提取了代码片段,但是如果将所有代码都放入一个文件中,它应该可以编译(或者需要稍作努力才能成功编译)。 当初始挑战的值为
"c100b894-1729-464d-ace1-52dbce11db42"
而 api 秘密是
"7zxMEF5p/Z8l2p2U7Ghv6x14Af+Fx+92tPgUdVQ748FOIrEoT9bgT+bTRfXc5pz8na+hL/QdrCVG7bh9KpT0eMTm"
HMAC 摘要后面的行应该是带符号的输出,我希望它是
"4JEpF3ix66GA2B+ooK128Ift4XQVtc137N9yeg4Kqsn9PI0Kpzbysl9M1IeCEdjg0zl00wkVqcsnG4bm nlMb3A=="
而实际上是
"336e394b567a55634d46376478344b594354767267636d39456f584f51326c376f334f2f3348796f6939647a7a516a456e41786c3551537541453930422f424b".
更麻烦的是,我可以使用 python 和 C# 复制正确的结果,非常简单地使用库函数,我很不确定我哪里出错了。
您似乎过分喜欢对数据进行十六进制编码!
首先,在您的 sha256
函数中,您正确地散列数据以获得 32 字节的摘要,但是随后您对其进行十六进制编码以获得您稍后使用的 64 个十六进制字符(加上空终止符)作为 HMAC 的输入。您需要使用那些原始的 32 个字节。
然后,在计算 HMAC 并对结果进行 base 64 编码之后,在打印出来之前对 that 进行十六进制编码。没有必要这样做,base 64 已经包含可打印字符。
去掉你进行十六进制编码的那两个循环(并改变 sha256
所以你 return 正确的缓冲区)它应该可以正常工作。