2018-12-31 22:04:05 +03:00
|
|
|
//
|
2021-01-01 15:57:46 +03:00
|
|
|
// Copyright Aliaksei Levin (levlam@telegram.org), Arseny Smirnov (arseny30@gmail.com) 2014-2021
|
2018-12-31 22:04:05 +03:00
|
|
|
//
|
|
|
|
// Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
|
|
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
|
|
//
|
|
|
|
#include "td/utils/base64.h"
|
2019-07-06 13:29:15 +02:00
|
|
|
#include "td/utils/benchmark.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/utils/common.h"
|
|
|
|
#include "td/utils/crypto.h"
|
2020-06-12 19:43:58 +03:00
|
|
|
#include "td/utils/Random.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/utils/Slice.h"
|
2021-05-17 15:21:11 +03:00
|
|
|
#include "td/utils/SliceBuilder.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
#include "td/utils/tests.h"
|
2018-12-20 00:18:53 +03:00
|
|
|
#include "td/utils/UInt.h"
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
#include <limits>
|
|
|
|
|
|
|
|
static td::vector<td::string> strings{"", "1", "short test string", td::string(1000000, 'a')};
|
|
|
|
|
|
|
|
#if TD_HAVE_OPENSSL
|
2020-06-12 19:43:58 +03:00
|
|
|
#if TD_HAVE_ZLIB
|
2020-06-12 18:06:40 +03:00
|
|
|
TEST(Crypto, Aes) {
|
|
|
|
td::Random::Xorshift128plus rnd(123);
|
|
|
|
td::UInt256 key;
|
|
|
|
rnd.bytes(as_slice(key));
|
2020-06-12 19:43:58 +03:00
|
|
|
td::string plaintext(16, '\0');
|
|
|
|
td::string encrypted(16, '\0');
|
|
|
|
td::string decrypted(16, '\0');
|
2020-06-12 18:06:40 +03:00
|
|
|
rnd.bytes(plaintext);
|
|
|
|
|
|
|
|
td::AesState encryptor;
|
|
|
|
encryptor.init(as_slice(key), true);
|
|
|
|
td::AesState decryptor;
|
|
|
|
decryptor.init(as_slice(key), false);
|
|
|
|
|
2020-06-12 19:40:17 +03:00
|
|
|
encryptor.encrypt(td::as_slice(plaintext).ubegin(), td::as_slice(encrypted).ubegin(), 16);
|
|
|
|
decryptor.decrypt(td::as_slice(encrypted).ubegin(), td::as_slice(decrypted).ubegin(), 16);
|
2020-06-12 18:06:40 +03:00
|
|
|
|
|
|
|
CHECK(decrypted == plaintext);
|
|
|
|
CHECK(decrypted != encrypted);
|
|
|
|
CHECK(td::crc32(encrypted) == 178892237);
|
|
|
|
}
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
TEST(Crypto, AesCtrState) {
|
|
|
|
td::vector<td::uint32> answers1{0u, 1141589763u, 596296607u, 3673001485u, 2302125528u,
|
|
|
|
330967191u, 2047392231u, 3537459563u, 307747798u, 2149598133u};
|
|
|
|
td::vector<td::uint32> answers2{0u, 2053451992u, 1384063362u, 3266188502u, 2893295118u,
|
|
|
|
780356167u, 1904947434u, 2043402406u, 472080809u, 1807109488u};
|
|
|
|
|
|
|
|
std::size_t i = 0;
|
|
|
|
for (auto length : {0, 1, 31, 32, 33, 9999, 10000, 10001, 999999, 1000001}) {
|
|
|
|
td::uint32 seed = length;
|
|
|
|
td::string s(length, '\0');
|
|
|
|
for (auto &c : s) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = static_cast<char>((seed >> 23) & 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
td::UInt256 key;
|
|
|
|
for (auto &c : key.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
td::UInt128 iv;
|
|
|
|
for (auto &c : iv.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
td::AesCtrState state;
|
2019-08-12 14:45:57 +03:00
|
|
|
state.init(as_slice(key), as_slice(iv));
|
2018-12-31 22:04:05 +03:00
|
|
|
td::string t(length, '\0');
|
2020-09-27 22:07:42 +03:00
|
|
|
std::size_t pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length, '\0'))) {
|
|
|
|
auto len = str.size();
|
|
|
|
state.encrypt(td::Slice(s).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
ASSERT_EQ(answers1[i], td::crc32(t));
|
2019-08-12 14:45:57 +03:00
|
|
|
state.init(as_slice(key), as_slice(iv));
|
2020-09-27 22:07:42 +03:00
|
|
|
pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length, '\0'))) {
|
|
|
|
auto len = str.size();
|
|
|
|
state.decrypt(td::Slice(t).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2020-06-15 23:20:44 +03:00
|
|
|
ASSERT_STREQ(td::base64_encode(s), td::base64_encode(t));
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
for (auto &c : iv.raw) {
|
|
|
|
c = 0xFF;
|
|
|
|
}
|
2019-08-12 14:45:57 +03:00
|
|
|
state.init(as_slice(key), as_slice(iv));
|
2020-09-27 22:07:42 +03:00
|
|
|
pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length, '\0'))) {
|
|
|
|
auto len = str.size();
|
|
|
|
state.encrypt(td::Slice(s).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
ASSERT_EQ(answers2[i], td::crc32(t));
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-15 15:07:58 +03:00
|
|
|
TEST(Crypto, AesIgeState) {
|
2020-09-27 22:03:00 +03:00
|
|
|
td::vector<td::uint32> answers1{0u, 2045698207u, 2423540300u, 525522475u, 1545267325u, 724143417u};
|
2020-06-15 15:07:58 +03:00
|
|
|
|
|
|
|
std::size_t i = 0;
|
2020-09-27 22:03:00 +03:00
|
|
|
for (auto length : {0, 16, 32, 256, 1024, 65536}) {
|
2020-06-15 15:07:58 +03:00
|
|
|
td::uint32 seed = length;
|
|
|
|
td::string s(length, '\0');
|
|
|
|
for (auto &c : s) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = static_cast<char>((seed >> 23) & 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
td::UInt256 key;
|
|
|
|
for (auto &c : key.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
td::UInt256 iv;
|
|
|
|
for (auto &c : iv.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
td::AesIgeState state;
|
|
|
|
state.init(as_slice(key), as_slice(iv), true);
|
|
|
|
td::string t(length, '\0');
|
2020-09-27 22:03:00 +03:00
|
|
|
td::UInt256 iv_copy = iv;
|
|
|
|
td::string u(length, '\0');
|
|
|
|
std::size_t pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length / 16, '\0'))) {
|
|
|
|
auto len = 16 * str.size();
|
|
|
|
state.encrypt(td::Slice(s).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
td::aes_ige_encrypt(as_slice(key), as_slice(iv_copy), td::Slice(s).substr(pos, len),
|
|
|
|
td::MutableSlice(u).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2020-06-15 15:07:58 +03:00
|
|
|
|
|
|
|
ASSERT_EQ(answers1[i], td::crc32(t));
|
2020-09-27 22:03:00 +03:00
|
|
|
ASSERT_EQ(answers1[i], td::crc32(u));
|
2020-06-15 15:07:58 +03:00
|
|
|
|
|
|
|
state.init(as_slice(key), as_slice(iv), false);
|
2020-09-27 22:03:00 +03:00
|
|
|
iv_copy = iv;
|
|
|
|
pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length / 16, '\0'))) {
|
|
|
|
auto len = 16 * str.size();
|
|
|
|
state.decrypt(td::Slice(t).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
td::aes_ige_decrypt(as_slice(key), as_slice(iv_copy), td::Slice(u).substr(pos, len),
|
|
|
|
td::MutableSlice(u).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2020-06-15 23:20:44 +03:00
|
|
|
ASSERT_STREQ(td::base64_encode(s), td::base64_encode(t));
|
2020-09-27 22:03:00 +03:00
|
|
|
ASSERT_STREQ(td::base64_encode(s), td::base64_encode(u));
|
|
|
|
|
2020-06-15 15:07:58 +03:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2020-06-16 18:43:49 +03:00
|
|
|
|
|
|
|
TEST(Crypto, AesCbcState) {
|
2020-09-27 21:03:05 +03:00
|
|
|
td::vector<td::uint32> answers1{0u, 3617355989u, 3449188102u, 186999968u, 4244808847u, 2626031206u};
|
2020-06-16 18:43:49 +03:00
|
|
|
|
|
|
|
std::size_t i = 0;
|
2020-09-27 21:03:05 +03:00
|
|
|
for (auto length : {0, 16, 32, 256, 1024, 65536}) {
|
2020-06-16 18:43:49 +03:00
|
|
|
td::uint32 seed = length;
|
|
|
|
td::string s(length, '\0');
|
|
|
|
for (auto &c : s) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = static_cast<char>((seed >> 23) & 255);
|
|
|
|
}
|
|
|
|
|
|
|
|
td::UInt256 key;
|
|
|
|
for (auto &c : key.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
td::UInt128 iv;
|
|
|
|
for (auto &c : iv.raw) {
|
|
|
|
seed = seed * 123457567u + 987651241u;
|
|
|
|
c = (seed >> 23) & 255;
|
|
|
|
}
|
|
|
|
|
|
|
|
td::AesCbcState state(as_slice(key), as_slice(iv));
|
|
|
|
td::string t(length, '\0');
|
2020-09-27 21:03:05 +03:00
|
|
|
td::UInt128 iv_copy = iv;
|
|
|
|
td::string u(length, '\0');
|
|
|
|
std::size_t pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length / 16, '\0'))) {
|
|
|
|
auto len = 16 * str.size();
|
|
|
|
state.encrypt(td::Slice(s).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
td::aes_cbc_encrypt(as_slice(key), as_slice(iv_copy), td::Slice(s).substr(pos, len),
|
|
|
|
td::MutableSlice(u).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2020-06-16 18:43:49 +03:00
|
|
|
|
|
|
|
ASSERT_EQ(answers1[i], td::crc32(t));
|
2020-09-27 21:03:05 +03:00
|
|
|
ASSERT_EQ(answers1[i], td::crc32(u));
|
2020-06-16 18:43:49 +03:00
|
|
|
|
|
|
|
state = td::AesCbcState(as_slice(key), as_slice(iv));
|
2020-09-27 21:03:05 +03:00
|
|
|
iv_copy = iv;
|
|
|
|
pos = 0;
|
|
|
|
for (auto str : td::rand_split(td::string(length / 16, '\0'))) {
|
|
|
|
auto len = 16 * str.size();
|
|
|
|
state.decrypt(td::Slice(t).substr(pos, len), td::MutableSlice(t).substr(pos, len));
|
|
|
|
td::aes_cbc_decrypt(as_slice(key), as_slice(iv_copy), td::Slice(u).substr(pos, len),
|
|
|
|
td::MutableSlice(u).substr(pos, len));
|
|
|
|
pos += len;
|
|
|
|
}
|
2020-06-16 18:43:49 +03:00
|
|
|
ASSERT_STREQ(td::base64_encode(s), td::base64_encode(t));
|
2020-09-27 21:03:05 +03:00
|
|
|
ASSERT_STREQ(td::base64_encode(s), td::base64_encode(u));
|
|
|
|
|
2020-06-16 18:43:49 +03:00
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
2020-06-15 23:20:44 +03:00
|
|
|
#endif
|
2020-06-15 15:07:58 +03:00
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
TEST(Crypto, Sha256State) {
|
|
|
|
for (auto length : {0, 1, 31, 32, 33, 9999, 10000, 10001, 999999, 1000001}) {
|
|
|
|
auto s = td::rand_string(std::numeric_limits<char>::min(), std::numeric_limits<char>::max(), length);
|
|
|
|
td::UInt256 baseline;
|
2018-11-06 19:00:03 +03:00
|
|
|
td::sha256(s, as_slice(baseline));
|
2018-12-31 22:04:05 +03:00
|
|
|
|
|
|
|
td::Sha256State state;
|
2019-07-23 03:20:11 +03:00
|
|
|
state.init();
|
2019-07-23 03:26:26 +03:00
|
|
|
td::Sha256State state2 = std::move(state);
|
2018-12-31 22:04:05 +03:00
|
|
|
auto v = td::rand_split(s);
|
|
|
|
for (auto &x : v) {
|
2019-07-23 03:26:26 +03:00
|
|
|
state2.feed(x);
|
2018-12-31 22:04:05 +03:00
|
|
|
}
|
2019-07-23 03:26:26 +03:00
|
|
|
state = std::move(state2);
|
2018-12-31 22:04:05 +03:00
|
|
|
td::UInt256 result;
|
2019-07-23 03:20:11 +03:00
|
|
|
state.extract(as_slice(result));
|
2018-12-31 22:04:05 +03:00
|
|
|
ASSERT_TRUE(baseline == result);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Crypto, PBKDF) {
|
2020-06-12 19:43:58 +03:00
|
|
|
td::vector<td::string> passwords{"", "qwerty", td::string(1000, 'a')};
|
|
|
|
td::vector<td::string> salts{"", "qwerty", td::string(1000, 'a')};
|
2018-12-31 22:04:05 +03:00
|
|
|
td::vector<int> iteration_counts{1, 2, 1000};
|
|
|
|
td::vector<td::Slice> answers{
|
|
|
|
"984LZT0tcqQQjPWr6RL/3Xd2Ftu7J6cOggTzri0Pb60=", "lzmEEdaupDp3rO+SImq4J41NsGaL0denanJfdoCsRcU=",
|
|
|
|
"T8WKIcEAzhg1uPmZHXOLVpZdFLJOF2H73/xprF4LZno=", "NHxAnMhPOATsb1wV0cGDlAIs+ofzI6I4I8eGJeWN9Qw=",
|
|
|
|
"fjYi7waEPjbVYEuZ61/Nm2hbk/vRdShoJoXg4Ygnqe4=", "GhW6e95hGJSf+ID5IrSbvzWyBZ1l35A+UoL55Uh/njk=",
|
|
|
|
"BueLDpqSCEc0GWk83WgMwz3UsWwfvVKcvllETSB/Yq8=", "hgHgJZNWRh78PyPdVJsK8whgHOHQbNQiyaTuGDX2IFo=",
|
|
|
|
"T2xdyNT1GlcA4+MVNzOe7NCgSAAzNkanNsmuoSr+4xQ=", "/f6t++GUPE+e63+0TrlInL+UsmzRSAAFopa8BBBmb2w=",
|
|
|
|
"8Zn98QEAKS9wPOUlN09+pfm0SWs1IGeQxQkNMT/1k48=", "sURLQ/6UX/KVYedyQB21oAtMJ+STZ4iwpxfQtqmWkLw=",
|
|
|
|
"T9t/EJXFpPs2Lhca7IVGphTC/OdEloPMHw1UhDnXcyQ=", "TIrtN05E9KQL6Lp/wjtbsFS+KkWZ8jlGK0ErtaoitOg=",
|
|
|
|
"+1KcMBjyUNz5VMaIfE5wkGwS6I+IQ5FhK+Ou2HgtVoQ=", "h36ci1T0vGllCl/xJxq6vI7n28Bg40dilzWOKg6Jt8k=",
|
|
|
|
"9uwsHJsotTiTqqCYftN729Dg7QI2BijIjV2MvSEUAeE=", "/l+vd/XYgbioh1SfLMaGRr13udmY6TLSlG4OYmytwGU=",
|
|
|
|
"7qfZZBbMRLtgjqq7GHgWa/UfXPajW8NXpJ6/T3P1rxI=", "ufwz94p28WnoOFdbrb1oyQEzm/v0CV2b0xBVxeEPJGA=",
|
|
|
|
"T/PUUBX2vGMUsI6httlhbMHlGPMvqFBNzayU5voVlaw=", "viMvsvTg9GfQymF3AXZ8uFYTDa3qLrqJJk9w/74iZfg=",
|
|
|
|
"HQF+rOZMW4DAdgZz8kAMe28eyIi0rs3a3u/mUeGPNfs=", "7lBVA+GnSxWF/eOo+tyyTB7niMDl1MqP8yzo+xnHTyw=",
|
|
|
|
"aTWb7HQAxaTKhSiRPY3GuM1GVmq/FPuwWBU/TUpdy70=", "fbg8M/+Ht/oU+UAZ4dQcGPo+wgCCHaA+GM4tm5jnWcY=",
|
|
|
|
"DJbCGFMIR/5neAlpda8Td5zftK4NGekVrg2xjrKW/4c="};
|
|
|
|
|
|
|
|
std::size_t pos = 0;
|
|
|
|
for (auto &password : passwords) {
|
|
|
|
for (auto &salt : salts) {
|
|
|
|
for (auto &iteration_count : iteration_counts) {
|
|
|
|
char result[32];
|
|
|
|
td::pbkdf2_sha256(password, salt, iteration_count, {result, 32});
|
|
|
|
ASSERT_STREQ(answers[pos], td::base64_encode({result, 32}));
|
|
|
|
pos++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Crypto, sha1) {
|
|
|
|
td::vector<td::Slice> answers{"2jmj7l5rSw0yVb/vlWAYkK/YBwk=", "NWoZK3kTsExUV00Ywo1G5jlUKKs=",
|
|
|
|
"uRysQwoax0pNJeBC3+zpQzJy1rA=", "NKqXPNTE2qT2Husr260nMWU0AW8="};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
unsigned char output[20];
|
|
|
|
td::sha1(strings[i], output);
|
|
|
|
ASSERT_STREQ(answers[i], td::base64_encode(td::Slice(output, 20)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Crypto, sha256) {
|
|
|
|
td::vector<td::Slice> answers{
|
|
|
|
"47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU=", "a4ayc/80/OGda4BO/1o/V0etpOqiLx1JwB5S3beHW0s=",
|
|
|
|
"yPMaY7Q8PKPwCsw64UnDD5mhRcituEJgzLZMvr0O8pY=", "zcduXJkU+5KBocfihNc+Z/GAmkiklyAOBG05zMcRLNA="};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
td::string output(32, '\0');
|
|
|
|
td::sha256(strings[i], output);
|
|
|
|
ASSERT_STREQ(answers[i], td::base64_encode(output));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Crypto, md5) {
|
|
|
|
td::vector<td::Slice> answers{
|
|
|
|
"1B2M2Y8AsgTpgAmY7PhCfg==", "xMpCOKC5I4INzFCab3WEmw==", "vwBninYbDRkgk+uA7GMiIQ==", "dwfWrk4CfHDuoqk1wilvIQ=="};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
td::string output(16, '\0');
|
|
|
|
td::md5(strings[i], output);
|
|
|
|
ASSERT_STREQ(answers[i], td::base64_encode(output));
|
|
|
|
}
|
|
|
|
}
|
2020-12-22 17:54:35 +03:00
|
|
|
|
|
|
|
TEST(Crypto, hmac_sha256) {
|
|
|
|
td::vector<td::Slice> answers{
|
|
|
|
"t33rfT85UOe6N00BhsNwobE+f2TnW331HhdvQ4GdJp8=", "BQl5HF2jqhCz4JTqhAs+H364oxboh7QlluOMHuuRVh8=",
|
|
|
|
"NCCPuZBsAPBd/qr3SyeYE+e1RNgzkKJCS/+eXDBw8zU=", "mo3ahTkyLKfoQoYA0s7vRZULuH++vqwFJD0U5n9HHw0="};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
td::string output(32, '\0');
|
|
|
|
td::hmac_sha256("cucumber", strings[i], output);
|
|
|
|
ASSERT_STREQ(answers[i], td::base64_encode(output));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(Crypto, hmac_sha512) {
|
|
|
|
td::vector<td::Slice> answers{
|
|
|
|
"o28hTN1m/TGlm/VYxDIzOdUE4wMpQzO8hVcTkiP2ezEJXtrOvCjRnl20aOV1S8axA5Te0TzIjfIoEAtpzamIsA==",
|
|
|
|
"32X3GslSz0HDznSrCNt++ePRcFVSUSD+tfOVannyxS+yLt/om11qILCE64RFTS8/B84gByMzC3FuAlfcIam/KA==",
|
|
|
|
"BVqe5rK1Fg1i+C7xXTAzT9vDPcf3kQQpTtse6rT/EVDzKo9AUo4ZwyUyJ0KcLHoffIjul/TuJoBg+wLz7Z7r7g==",
|
|
|
|
"WASmeku5Pcfz7N0Kp4Q3I9sxtO2MiaBXA418CY0HvjdtmAo7QY+K3E0o9UemgGzz41KqeypzRC92MwOAOnXJLA=="};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
td::string output(64, '\0');
|
|
|
|
td::hmac_sha512("cucumber", strings[i], output);
|
|
|
|
ASSERT_STREQ(answers[i], td::base64_encode(output));
|
|
|
|
}
|
|
|
|
}
|
2018-12-31 22:04:05 +03:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if TD_HAVE_ZLIB
|
|
|
|
TEST(Crypto, crc32) {
|
|
|
|
td::vector<td::uint32> answers{0u, 2212294583u, 3013144151u, 3693461436u};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
ASSERT_EQ(answers[i], td::crc32(strings[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-12-19 17:48:39 +03:00
|
|
|
#if TD_HAVE_CRC32C
|
|
|
|
TEST(Crypto, crc32c) {
|
|
|
|
td::vector<td::uint32> answers{0u, 2432014819u, 1077264849u, 1131405888u};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
ASSERT_EQ(answers[i], td::crc32c(strings[i]));
|
2019-07-06 13:29:15 +02:00
|
|
|
|
|
|
|
auto v = td::rand_split(strings[i]);
|
|
|
|
td::uint32 a = 0;
|
|
|
|
td::uint32 b = 0;
|
|
|
|
for (auto &x : v) {
|
|
|
|
a = td::crc32c_extend(a, x);
|
|
|
|
auto x_crc = td::crc32c(x);
|
|
|
|
b = td::crc32c_extend(b, x_crc, x.size());
|
|
|
|
}
|
|
|
|
ASSERT_EQ(answers[i], a);
|
|
|
|
ASSERT_EQ(answers[i], b);
|
2018-12-19 17:48:39 +03:00
|
|
|
}
|
|
|
|
}
|
2019-07-21 21:07:07 +03:00
|
|
|
|
2019-07-06 13:29:15 +02:00
|
|
|
TEST(Crypto, crc32c_benchmark) {
|
|
|
|
class Crc32cExtendBenchmark : public td::Benchmark {
|
|
|
|
public:
|
2019-07-21 21:07:07 +03:00
|
|
|
explicit Crc32cExtendBenchmark(size_t chunk_size) : chunk_size_(chunk_size) {
|
2019-07-06 13:29:15 +02:00
|
|
|
}
|
2020-06-12 19:43:58 +03:00
|
|
|
td::string get_description() const override {
|
2019-07-06 13:29:15 +02:00
|
|
|
return PSTRING() << "Crc32c with chunk_size=" << chunk_size_;
|
|
|
|
}
|
|
|
|
void start_up_n(int n) override {
|
|
|
|
if (n > (1 << 20)) {
|
|
|
|
cnt_ = n / (1 << 20);
|
|
|
|
n = (1 << 20);
|
|
|
|
} else {
|
|
|
|
cnt_ = 1;
|
|
|
|
}
|
2020-06-12 19:43:58 +03:00
|
|
|
data_ = td::string(n, 'a');
|
2019-07-06 13:29:15 +02:00
|
|
|
}
|
|
|
|
void run(int n) override {
|
|
|
|
td::uint32 res = 0;
|
|
|
|
for (int i = 0; i < cnt_; i++) {
|
|
|
|
td::Slice data(data_);
|
|
|
|
while (!data.empty()) {
|
|
|
|
auto head = data.substr(0, chunk_size_);
|
|
|
|
data = data.substr(head.size());
|
|
|
|
res = td::crc32c_extend(res, head);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
td::do_not_optimize_away(res);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
size_t chunk_size_;
|
2020-06-12 19:43:58 +03:00
|
|
|
td::string data_;
|
2019-07-06 13:29:15 +02:00
|
|
|
int cnt_;
|
|
|
|
};
|
|
|
|
bench(Crc32cExtendBenchmark(2));
|
|
|
|
bench(Crc32cExtendBenchmark(8));
|
|
|
|
bench(Crc32cExtendBenchmark(32));
|
|
|
|
bench(Crc32cExtendBenchmark(128));
|
|
|
|
bench(Crc32cExtendBenchmark(65536));
|
|
|
|
}
|
2018-12-19 17:48:39 +03:00
|
|
|
#endif
|
|
|
|
|
2018-12-31 22:04:05 +03:00
|
|
|
TEST(Crypto, crc64) {
|
|
|
|
td::vector<td::uint64> answers{0ull, 3039664240384658157ull, 17549519902062861804ull, 8794730974279819706ull};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
ASSERT_EQ(answers[i], td::crc64(strings[i]));
|
|
|
|
}
|
|
|
|
}
|
2018-03-26 19:09:25 +03:00
|
|
|
|
2019-07-06 13:29:15 +02:00
|
|
|
TEST(Crypto, crc16) {
|
|
|
|
td::vector<td::uint16> answers{0, 9842, 25046, 37023};
|
|
|
|
|
|
|
|
for (std::size_t i = 0; i < strings.size(); i++) {
|
|
|
|
ASSERT_EQ(answers[i], td::crc16(strings[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-26 19:09:25 +03:00
|
|
|
static td::Slice rsa_private_key = R"ABCD(
|
|
|
|
-----BEGIN PRIVATE KEY-----
|
|
|
|
MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQDeYT5/prmLEa2Q
|
|
|
|
tZND+UwTmif8kl2VlXaMCjj1k1lJJq8BqS8cVM2vPnOPzFoiC2LYykhm4kk7goCC
|
|
|
|
ZH6wez9yakg28fcq0Ycv0x8DL1K+VKHJuwIhVfQs//IY1/cBOrMESc+NQowPbv1t
|
|
|
|
TIFxBO2gebnpLuseht8ix7XtpGC4qAaHN2aEvT2cRsnA76TAK1RVxf1OYGUFBDzY
|
|
|
|
318WpVZfVIjcQ7K9+eU6b2Yb84VLlvJXw3e1rvw+fBzx2EjpD4zhXy11YppWDyV6
|
|
|
|
HEb2hs3cGS/LbHfHvdcSfil2omaJP97MDEEY2HFxjR/E5CEf2suvPzX4XS3RE+S3
|
|
|
|
2aEJaaQbAgMBAAECggEAKo3XRNwls0wNt5xXcvF4smOUdUuY5u/0AHZQUgYBVvM1
|
|
|
|
GA9E+ZnsxjUgLgs/0DX3k16aHj39H4sohksuxxy+lmlqKkGBN8tioC85RwW+Qre1
|
|
|
|
QgIsNS7ai+XqcQCavrx51z88nV53qNhnXIwAVR1JT6Ubg1i8G1pZxrEKyk/jRlJd
|
|
|
|
mGjf6vjitH//PPkghPJ/D42k93YRcy+duOgqYDQpLZp8DiEGfYrX10B1H7HrWLV+
|
|
|
|
Wp5KO1YXtKgQUplj6kYy72bVajbxYTvzgjaaKsh74jBO0uT3tHTtXG0dcKGb0VR/
|
|
|
|
cqP/1H/lC9bAnAqAGefNusGJQZIElvTsrpIQXOeZsQKBgQD2W04S+FjqYYFjnEFX
|
|
|
|
6eL4it01afs5M3/C6CcI5JQtN6p+Na4NCSILol33xwhakn87zqdADHawBYQVQ8Uw
|
|
|
|
dPurl805wfkzN3AbfdDmtx0IJ8vK4HFpktRjfpwBVhlVtm1doAYFqqsuCF2vWW1t
|
|
|
|
mM2YOSq4AnRHCeBb/P6kRIW0MwKBgQDnFawKKqiC4tuyBOkkEhexlm7x9he0md7D
|
|
|
|
3Z2hc3Bmdcq1niw4wBq3HUxGLReGCcSr5epKSQwkunlTn5ZSC6Rmbe4zxsGIwbb3
|
|
|
|
5W3342swBaoxEIuBokBvZ/xUOXVwiqKj+S/NzVkZcnT6K9V/HnUCQR+JBbQxFQaX
|
|
|
|
iiezcjKoeQKBgCIVUcDoIQ0UPl10ocmy7xbpx177calhSZzCl5vwW9vBptHdRV5C
|
|
|
|
VDZ92ThNjgdR205/8b23u7fwm2yBusdQd/0ufFMwVfTTB6yWBI/W56pYLya7VJWB
|
|
|
|
nebB/n1k1w53tbvNRugDy7kLqUJ4Qd521ILp7dIVbNbjM+omH2jEnibnAoGBAIM5
|
|
|
|
a1jaoJay/M86uqohHBNcuePtO8jzF+1iDAGC7HFCsrov+CzB6mnR2V6AfLtBEM4M
|
|
|
|
4d8NXDf/LKawGUy+D72a74m3dG+UkbJ0Nt5t5pB+pwb1vkL/QFgDVOb/OhGOqI01
|
|
|
|
FFBqLA6nUIZAHhzxzsBY+u90rb6xkey8J49faiUBAoGAaMgOgEvQB5H19ZL5tMkl
|
|
|
|
A/DKtTz/NFzN4Zw/vNPVb7eNn4jg9M25d9xqvL4acOa+nuV3nLHbcUWE1/7STXw1
|
|
|
|
gT58CvoEmD1AiP95nup+HKHENJ1DWMgF5MDfVQwGCvWP5/Qy89ybr0eG8HjbldbN
|
|
|
|
MpSmzz2wOz152oGdOd3syT4=
|
|
|
|
-----END PRIVATE KEY-----
|
|
|
|
)ABCD";
|
|
|
|
|
|
|
|
static td::Slice rsa_public_key = R"ABCD(
|
|
|
|
-----BEGIN PUBLIC KEY-----
|
|
|
|
MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA3mE+f6a5ixGtkLWTQ/lM
|
|
|
|
E5on/JJdlZV2jAo49ZNZSSavAakvHFTNrz5zj8xaIgti2MpIZuJJO4KAgmR+sHs/
|
|
|
|
cmpINvH3KtGHL9MfAy9SvlShybsCIVX0LP/yGNf3ATqzBEnPjUKMD279bUyBcQTt
|
|
|
|
oHm56S7rHobfIse17aRguKgGhzdmhL09nEbJwO+kwCtUVcX9TmBlBQQ82N9fFqVW
|
|
|
|
X1SI3EOyvfnlOm9mG/OFS5byV8N3ta78Pnwc8dhI6Q+M4V8tdWKaVg8lehxG9obN
|
|
|
|
3Bkvy2x3x73XEn4pdqJmiT/ezAxBGNhxcY0fxOQhH9rLrz81+F0t0RPkt9mhCWmk
|
|
|
|
GwIDAQAB
|
|
|
|
-----END PUBLIC KEY-----
|
|
|
|
)ABCD";
|
|
|
|
|
|
|
|
TEST(Crypto, rsa) {
|
|
|
|
auto value = td::rand_string('a', 'z', 200);
|
|
|
|
auto encrypted_value = td::rsa_encrypt_pkcs1_oaep(rsa_public_key, value).move_as_ok();
|
|
|
|
auto decrypted_value = td::rsa_decrypt_pkcs1_oaep(rsa_private_key, encrypted_value.as_slice()).move_as_ok();
|
|
|
|
ASSERT_TRUE(decrypted_value.as_slice().truncate(value.size()) == value);
|
|
|
|
}
|