/* This file is part of TON Blockchain Library. TON Blockchain Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. TON Blockchain Library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with TON Blockchain Library. If not, see . Copyright 2017-2019 Telegram Systems LLP */ #include "td/utils/base64.h" #include "td/utils/benchmark.h" #include "td/utils/common.h" #include "td/utils/crypto.h" #include "td/utils/logging.h" #include "td/utils/Slice.h" #include "td/utils/tests.h" #include "td/utils/UInt.h" #include static td::vector strings{"", "1", "short test string", td::string(1000000, 'a')}; #if TD_HAVE_OPENSSL TEST(Crypto, AesCtrState) { td::vector answers1{0u, 1141589763u, 596296607u, 3673001485u, 2302125528u, 330967191u, 2047392231u, 3537459563u, 307747798u, 2149598133u}; td::vector 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((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; state.init(as_slice(key), as_slice(iv)); td::string t(length, '\0'); state.encrypt(s, t); ASSERT_EQ(answers1[i], td::crc32(t)); state.init(as_slice(key), as_slice(iv)); state.decrypt(t, t); ASSERT_STREQ(s, t); for (auto &c : iv.raw) { c = 0xFF; } state.init(as_slice(key), as_slice(iv)); state.encrypt(s, t); ASSERT_EQ(answers2[i], td::crc32(t)); i++; } } TEST(Crypto, Sha256State) { for (auto length : {0, 1, 31, 32, 33, 9999, 10000, 10001, 999999, 1000001}) { auto s = td::rand_string(std::numeric_limits::min(), std::numeric_limits::max(), length); td::UInt256 baseline; td::sha256(s, as_slice(baseline)); td::Sha256State state; state.init(); td::Sha256State state2 = std::move(state); auto v = td::rand_split(s); for (auto &x : v) { state2.feed(x); } state = std::move(state2); td::UInt256 result; state.extract(as_slice(result)); ASSERT_TRUE(baseline == result); } } TEST(Crypto, PBKDF) { td::vector passwords{"", "qwerty", std::string(1000, 'a')}; td::vector salts{"", "qwerty", std::string(1000, 'a')}; td::vector iteration_counts{1, 2, 1000}; td::vector 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 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 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 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)); } } #endif #if TD_HAVE_ZLIB TEST(Crypto, crc32) { td::vector answers{0u, 2212294583u, 3013144151u, 3693461436u}; for (std::size_t i = 0; i < strings.size(); i++) { ASSERT_EQ(answers[i], td::crc32(strings[i])); } } #endif #if TD_HAVE_CRC32C TEST(Crypto, crc32c) { td::vector answers{0u, 2432014819u, 1077264849u, 1131405888u}; for (std::size_t i = 0; i < strings.size(); i++) { ASSERT_EQ(answers[i], td::crc32c(strings[i])); 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); } } TEST(Crypto, crc32c_benchmark) { class Crc32cExtendBenchmark : public td::Benchmark { public: explicit Crc32cExtendBenchmark(size_t chunk_size) : chunk_size_(chunk_size) { } std::string get_description() const override { 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; } data_ = std::string(n, 'a'); } 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_; std::string data_; int cnt_; }; bench(Crc32cExtendBenchmark(2)); bench(Crc32cExtendBenchmark(8)); bench(Crc32cExtendBenchmark(32)); bench(Crc32cExtendBenchmark(128)); bench(Crc32cExtendBenchmark(65536)); } #endif TEST(Crypto, crc64) { td::vector answers{0ull, 3039664240384658157ull, 17549519902062861804ull, 8794730974279819706ull}; for (std::size_t i = 0; i < strings.size(); i++) { ASSERT_EQ(answers[i], td::crc64(strings[i])); } } TEST(Crypto, crc16) { td::vector answers{0, 9842, 25046, 37023}; for (std::size_t i = 0; i < strings.size(); i++) { ASSERT_EQ(answers[i], td::crc16(strings[i])); } } 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); }