-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathhmac.hpp
129 lines (121 loc) · 5.21 KB
/
hmac.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
#ifndef HMAC_HPP_INCLUDED
#define HMAC_HPP_INCLUDED
#include "sha256.hpp"
#include "sha512.hpp"
#include <string>
namespace hmac {
/** \brief Тип используемой хеш-функции
*/
enum class TypeHash {
SHA256 = 0, /**< Использовать SHA256 */
SHA512 = 1, /**< Использовать SHA512 */
};
/** \brief Преобразование строки в шестнадцетиричный формат
* \param input Исходная строка
* \param is_upper Флаг верхнего регистра
* \return Строка в шестнадцетиричном формате
*/
std::string to_hex(const std::string& input, const bool is_upper = false) {
static const char *lut = "0123456789abcdef0123456789ABCDEF";
const char *symbol = is_upper ? lut + 16 : lut;
const size_t length = input.size();
std::string output;
output.reserve(2 * length);
for (size_t i = 0; i < length; ++i) {
const uint8_t c = input[i];
output.push_back(symbol[c >> 4]);
output.push_back(symbol[c & 15]);
}
return output;
}
/** \brief Получить хеш строки
* \param input Исходная строка
* \param type Тип хеш-функции
* \return Хеш строки
*/
std::string get_hash(const std::string &input, const TypeHash type) {
switch(type) {
case TypeHash::SHA256: {
uint8_t digest[hmac_hash::SHA256::DIGEST_SIZE];
std::fill(digest, digest + hmac_hash::SHA256::DIGEST_SIZE, '\0');
hmac_hash::SHA256 ctx = hmac_hash::SHA256();
ctx.init();
ctx.update((uint8_t*)input.c_str(), input.size());
ctx.final(digest);
return std::string((const char*)digest, hmac_hash::SHA256::DIGEST_SIZE);
}
break;
case TypeHash::SHA512: {
uint8_t digest[hmac_hash::SHA512::DIGEST_SIZE];
std::fill(digest, digest + hmac_hash::SHA512::DIGEST_SIZE, '\0');
hmac_hash::SHA512 ctx = hmac_hash::SHA512();
ctx.init();
ctx.update((uint8_t*)input.c_str(), input.size());
ctx.final(digest);
return std::string((const char*)digest, hmac_hash::SHA512::DIGEST_SIZE);
}
break;
default:
return std::string();
break;
};
return std::string();
}
/** \brief Получить HMAC
*
* \param key Ключ
* \param msg Сообщение
* \param type Тип хеш-функции
* \param is_hex Использовать шестнадцетиричное представление
* \param is_upper Флаг верхнего регистра
*/
std::string get_hmac(std::string key, const std::string &msg, const TypeHash type, const bool is_hex = true, const bool is_upper = false) {
size_t block_size = 0;
switch(type) {
case TypeHash::SHA256:
block_size = hmac_hash::SHA256::SHA224_256_BLOCK_SIZE;
break;
case TypeHash::SHA512:
block_size = hmac_hash::SHA512::SHA384_512_BLOCK_SIZE;
break;
default:
return std::string();
break;
};
if(key.size() > block_size) {
/* Если длина ключа K больше размера блока,
* то к ключу K применяем хеш-функцию H,
* получаем строку размером в L байт,
* добавляем нули к правой части этой строки
* для создания строки размером в b байт,
*/
key = get_hash(key, type);
}
if(key.size() < block_size) {
/* Если длина ключа K меньше размера блока,
* то добавляем нули к правой части K
* для создания строки размером в b байт,
* копируем результат в K0
* (например, если length( К ) = 20 (в байтах)
* и b = 64 (в байтах), то к правой части К
* будет добавлено 64 - 20 = 44 нулевых байта (0x00))
*/
key.resize(block_size, '\0');
}
/* Если длина ключа K равна размеру блока,
* то копируем K в K0 без изменений
* и переходим к шагу 2
*/
std::string ikeypad;
std::string okeypad;
ikeypad.reserve(block_size);
okeypad.reserve(block_size);
for(size_t i = 0; i < block_size; ++i) {
ikeypad.push_back('\x36' ^ key[i]);
okeypad.push_back('\x5c' ^ key[i]);
}
if(is_hex) return to_hex(get_hash(okeypad + get_hash(ikeypad + msg, type), type), is_upper);
return get_hash(okeypad + get_hash(ikeypad + msg, type), type);
}
}
#endif // HMAC_H_INCLUDED