2021-06-18 17:28:49 -04:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2021, Peter Bocan <me@pbocan.net>
|
2024-12-25 16:24:43 -05:00
|
|
|
* Copyright (c) 2025, Altomani Gianluca <altomanigianluca@gmail.com>
|
2021-06-18 17:28:49 -04:00
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
*/
|
|
|
|
|
2024-03-13 07:30:59 -04:00
|
|
|
#include <LibCrypto/ASN1/PEM.h>
|
2024-11-29 14:58:33 -05:00
|
|
|
#include <LibCrypto/Certificate/Certificate.h>
|
2021-06-18 17:28:49 -04:00
|
|
|
#include <LibCrypto/Hash/SHA2.h>
|
2024-03-22 05:17:31 -04:00
|
|
|
#include <LibCrypto/PK/PK.h>
|
2021-06-18 17:28:49 -04:00
|
|
|
#include <LibCrypto/PK/RSA.h>
|
|
|
|
#include <LibTest/TestCase.h>
|
|
|
|
#include <cstring>
|
|
|
|
|
2022-04-01 13:58:27 -04:00
|
|
|
static ByteBuffer operator""_b(char const* string, size_t length)
|
2021-06-18 17:28:49 -04:00
|
|
|
{
|
2021-09-05 18:59:52 -04:00
|
|
|
return ByteBuffer::copy(string, length).release_value();
|
2021-06-18 17:28:49 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST_CASE(test_RSA_raw_encrypt)
|
|
|
|
{
|
|
|
|
ByteBuffer data { "hellohellohellohellohellohellohellohellohellohellohellohello123-"_b };
|
|
|
|
u8 result[] { 0x6f, 0x7b, 0xe2, 0xd3, 0x95, 0xf8, 0x8d, 0x87, 0x6d, 0x10, 0x5e, 0xc3, 0xcd, 0xf7, 0xbb, 0xa6, 0x62, 0x8e, 0x45, 0xa0, 0xf1, 0xe5, 0x0f, 0xdf, 0x69, 0xcb, 0xb6, 0xd5, 0x42, 0x06, 0x7d, 0x72, 0xa9, 0x5e, 0xae, 0xbf, 0xbf, 0x0f, 0xe0, 0xeb, 0x31, 0x31, 0xca, 0x8a, 0x81, 0x1e, 0xb9, 0xec, 0x6d, 0xcc, 0xb8, 0xa4, 0xac, 0xa3, 0x31, 0x05, 0xa9, 0xac, 0xc9, 0xd3, 0xe6, 0x2a, 0x18, 0xfe };
|
2024-12-15 10:13:31 -05:00
|
|
|
Crypto::PK::RSA rsa(Crypto::PK::RSAPublicKey<> {
|
2021-06-18 17:28:49 -04:00
|
|
|
"8126832723025844890518845777858816391166654950553329127845898924164623511718747856014227624997335860970996746552094406240834082304784428582653994490504519"_bigint,
|
2024-12-15 10:13:31 -05:00
|
|
|
"65537"_bigint,
|
|
|
|
});
|
2024-12-25 16:04:38 -05:00
|
|
|
auto buf = TRY_OR_FAIL(rsa.encrypt(data));
|
2021-06-18 17:28:49 -04:00
|
|
|
EXPECT(memcmp(result, buf.data(), buf.size()) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// RSA PKCS #1 1.5
|
|
|
|
TEST_CASE(test_RSA_PKCS_1_encrypt)
|
|
|
|
{
|
|
|
|
ByteBuffer data { "hellohellohellohellohellohellohellohellohello123-"_b };
|
2024-12-23 11:41:37 -05:00
|
|
|
|
|
|
|
auto keypair = TRY_OR_FAIL(Crypto::PK::RSA::generate_key_pair(1024));
|
|
|
|
Crypto::PK::RSA_PKCS1_EME rsa(keypair);
|
2024-12-25 16:04:38 -05:00
|
|
|
auto enc = TRY_OR_FAIL(rsa.encrypt(data));
|
|
|
|
auto dec = TRY_OR_FAIL(rsa.decrypt(enc));
|
2021-06-18 17:28:49 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
EXPECT(memcmp(dec.data(), "hellohellohellohellohellohellohellohellohello123-", 49) == 0);
|
2021-06-18 17:28:49 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// RSA | ASN1 PKCS1 DER / PEM encoded Key import
|
|
|
|
TEST_CASE(test_RSA_ASN1_PKCS1_DER_PEM_parse)
|
|
|
|
{
|
|
|
|
auto privkey = R"(-----BEGIN RSA PRIVATE KEY-----
|
|
|
|
MIIBOgIBAAJBAJsrIYHxs1YL9tpfodaWs1lJoMdF4kgFisUFSj6nvBhJUlmBh607AlgTaX0E
|
|
|
|
DGPYycXYGZ2n6rqmms5lpDXBpUcCAwEAAQJAUNpPkmtEHDENxsoQBUXvXDYeXdePSiIBJhpU
|
|
|
|
joNOYoR5R9z5oX2cpcyykQ58FC2vKKg+x8N6xczG7qO95tw5UQIhAN354CP/FA+uTeJ6KJ+i
|
|
|
|
zCBCl58CjNCzO0s5HTc56el5AiEAsvPKXo5/9gS/S4UzDRP6abq7GreixTfjR8LXidk3FL8C
|
|
|
|
IQCTjYI861Y+hjMnlORkGSdvWlTHUj6gjEOh4TlWeJzQoQIgAxMZOQKtxCZUuxFwzRq4xLRG
|
|
|
|
nrDlBQpuxz7bwSyQO7UCIHrYMnDohgNbwtA5ZpW3H1cKKQQvueWm6sxW9P5sUrZ3
|
2022-07-11 13:32:29 -04:00
|
|
|
-----END RSA PRIVATE KEY-----)"sv;
|
2021-06-18 17:28:49 -04:00
|
|
|
|
|
|
|
Crypto::PK::RSA rsa(privkey);
|
|
|
|
if (rsa.public_key().public_exponent() != 65537) {
|
|
|
|
FAIL("Invalid public exponent");
|
|
|
|
}
|
|
|
|
if (rsa.private_key().private_exponent() != "4234603516465654167360850580101327813936403862038934287300450163438938741499875303761385527882335478349599685406941909381269804396099893549838642251053393"_bigint) {
|
|
|
|
FAIL("Invalid private exponent");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// RSA | ASN1 PKCS8 DER / PEM encoded Key import
|
|
|
|
TEST_CASE(test_RSA_ASN1_PKCS8_DER_PEM_parse)
|
|
|
|
{
|
|
|
|
auto privkey = R"(-----BEGIN PRIVATE KEY-----
|
|
|
|
MIIEvgIBADANBgkqhkiG9w0BAQEFAASCBKgwggSkAgEAAoIBAQC7ZBYaG9+CcJP7
|
|
|
|
WVFJRI/uw3hljc7WpzeYs8MN82/g9CG1gnEF3P3ZSBdWVr8gnbh05EsSGHKghIce
|
|
|
|
CB7DNrM5Ab0ru04CuODdPx56xCj+4MmzTc/aq79ntmOt131NGHgq9yVwfJqnSpyl
|
|
|
|
OoVw7j/Wg4ciwPDQaeLmD1BsE/W9UsF1km7DWasBpW5br82DpudKgJq2Ixf52+rY
|
|
|
|
TCkMgyWcetx4MfXll4y5ZVtJXCnHJfkCS64EaCqXmClP4ovOuHH4khJ3rW9j4yuL
|
|
|
|
e5ck3PSXOrtOTR43HZkCXzseCkbW7qKSmk/9ZreImOzOgu8vvw7ewLAQR9qYVS6X
|
|
|
|
PXY8IilDAgMBAAECggEBAIV3ld5mt90Z/exqA2Fh+fofMyNxyz5Lv2d9sZHAL5FT
|
|
|
|
kKbND18TtaIKnMSb6Gl8rKJk76slyo7Vlb8oHXEBBsm1mV0KfVenAlHS4QyjpmdT
|
|
|
|
B5Yz97VR2nQuDfUFpHNC2GQRv5LMzQIWPFfaxKxYpRNOfvOb5Gks4bTmd2tjFAYR
|
|
|
|
MCbHgPw1liKA9dYKk4NB0301EY05e4Zz8RjqYHkkmOPD7DnjFbHqcFUjVKK5E3vD
|
|
|
|
WjxNXUbiSudCCN7WLEOyeHZNd+l6kSAVxZuCAp0G3Da5ndXgIStcy4hYi/fL3XQQ
|
|
|
|
bNpxjfhsjlD3tdHNr3NNYDAqxcxpsyO1NCpCIW3ZVrECgYEA7l6gTZ3e9AiSNlMd
|
|
|
|
2O2vNnbQ6UZfsEfu2y7HmpCuNJkFkAnM/1h72Krejnn31rRuR6uCFn4YgQUN9Eq0
|
|
|
|
E1PJCtTay2ucZw5rqtkewT9QzXvVD9eiGM+MF89UzSCC+dOW0/odkD+xP2evnPvG
|
|
|
|
PbXztnuERC1pi0YWLj1YcsfsEX0CgYEAyUA2UtYjnvCcteIy+rURT0aoZ9tDMrG+
|
|
|
|
Es42EURVv1sduVdUst5R+bXx1aDzpCkcdni3TyxeosvTGAZngI3O8ghh1GV7NPZR
|
|
|
|
nkiPXjMnhL0Zf+X9gCA6TFANfPuWhMSGijYsCd46diKGDReGYUnmcN9XopeG1h6i
|
|
|
|
3JiOuVPAIb8CgYBmIcUtfGb6yHFdNV+kgrJ/84ivaqe1MBz3bKO5ZiQ+BRKNFKXx
|
|
|
|
AkiOHSgeg8PdCpH1w1aJrJ1zKmdANIHThiKtsWXNot3wig03tq+mvSox4Mz5bLrX
|
|
|
|
RpYP3ZXIDhYQVMhbKt9f3upi8FoeOQJHjp5Nob6aN5rxQaZfSYmMJHzRQQKBgQCO
|
|
|
|
ALwUGTtLNBYvlKtKEadkG8RKfAFfbOFkXZLy/hfPDRjdJY0DJTIMk+BPT+F6rPOD
|
|
|
|
eMxHllQ0ZMPPiP1RTT5/s4BsISsdhMy0dhiLbGbvF4s9nugPly3rmPTbgp6DkjQo
|
|
|
|
o+7RC7iOkO+rnzTXwxBSBpXMiUTAIx/hrdfPVxQT+wKBgCh7N3OLIOH6EWcW1fif
|
|
|
|
UoENh8rkt/kzm89G1JLwBhuBIBPXUEZt2dS/xSUempqVqFGONpP87gvqxkMTtgCA
|
|
|
|
73KXn/cxHWM2kmXyHA3kQlOYw6WHjpldQAxLE+TRHXO2JUtZ09Mu4rVXX7lmwbTm
|
|
|
|
l3vmuDEF3/Bo1C1HTg0xRV/l
|
2022-07-11 13:32:29 -04:00
|
|
|
-----END PRIVATE KEY-----)"sv;
|
2021-06-18 17:28:49 -04:00
|
|
|
|
|
|
|
Crypto::PK::RSA rsa(privkey);
|
|
|
|
if (rsa.public_key().public_exponent() != 65537) {
|
|
|
|
FAIL("Invalid public exponent");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rsa.private_key().private_exponent() != "16848664331299797559656678180469464902267415922431923391961407795209879741791261105581093539484181644099608161661780611501562625272630894063592208758992911105496755004417051031019663332258403844985328863382168329621318366311519850803972480500782200178279692319955495383119697563295214236936264406600739633470565823022975212999060908747002623721589308539473108154612454595201561671949550531384574873324370774408913092560971930541734744950937900805812300970883306404011323308000168926094053141613790857814489531436452649384151085451448183385611208320292948291211969430321231180227006521681776197974694030147965578466993"_bigint) {
|
|
|
|
FAIL("Invalid private exponent");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-13 07:30:59 -04:00
|
|
|
TEST_CASE(test_RSA_keygen_enc)
|
|
|
|
{
|
|
|
|
auto keypem = R"(-----BEGIN PRIVATE KEY-----
|
|
|
|
MIIBVQIBADANBgkqhkiG9w0BAQEFAASCAT8wggE7AgEAAkEA5HMXMnY+RhEcYXsa
|
|
|
|
OyB/YkcrO1nxIeyDCMqwg5MDrSXO8vPXSEb9AZUNMF1jKiFWPoHxZ+foRxrLv4d9
|
|
|
|
sV/ETwIDAQABAkBpC37UJkjWQRHyxP83xuasExuO6/mT5sQN692kcppTJ9wHNWoD
|
|
|
|
9ZcREk4GGiklu4qx48/fYt8Cv6z6JuQ0ZQExAiEA9XRZVUnCJ2xOcCFCbyIF+d3F
|
|
|
|
9Kht5rR77F9KsRlgUbkCIQDuQ7YzLpQ8V8BJwKbDeXw1vQvcPEnyKnTOoALpF6bq
|
|
|
|
RwIhAIDSm8Ajgf7m3RQEoLVrCe/l8WtCqsuWliOsr6rbQq4hAiEAx8R16wvOtZlN
|
|
|
|
W4jvSU1+WwAaBZl21lfKf8OhLRXrmNkCIG9IRdcSiNR/Ut8QfD3N9Bb1HsUm+Bvz
|
|
|
|
c8yGzl89pYST
|
2024-03-22 05:17:31 -04:00
|
|
|
-----END PRIVATE KEY-----
|
|
|
|
)"sv;
|
2024-03-13 07:30:59 -04:00
|
|
|
auto decoded = Crypto::decode_pem(keypem.bytes());
|
2024-11-29 14:58:33 -05:00
|
|
|
EXPECT_EQ(decoded.type, Crypto::PEMType::PrivateKey);
|
|
|
|
auto decoder = Crypto::ASN1::Decoder { decoded.data };
|
|
|
|
auto priv_key_info = MUST(Crypto::Certificate::parse_private_key_info(decoder, {}));
|
2024-11-29 12:28:18 -05:00
|
|
|
auto keypair = MUST(Crypto::PK::RSA::parse_rsa_key(priv_key_info.raw_key, true, {}));
|
2024-11-29 14:58:33 -05:00
|
|
|
auto priv_der = MUST(priv_key_info.rsa.export_as_der());
|
2024-03-22 05:17:31 -04:00
|
|
|
auto rsa_encryption_oid = Array<int, 7> { 1, 2, 840, 113549, 1, 1, 1 };
|
2024-11-29 14:58:33 -05:00
|
|
|
auto wrapped_priv_der = MUST(Crypto::PK::wrap_in_private_key_info(priv_key_info.raw_key, rsa_encryption_oid, nullptr));
|
2024-03-22 05:17:31 -04:00
|
|
|
auto priv_pem = MUST(Crypto::encode_pem(wrapped_priv_der, Crypto::PEMType::PrivateKey));
|
2024-03-13 07:30:59 -04:00
|
|
|
auto rsa_from_pair = Crypto::PK::RSA(keypair.public_key, keypair.private_key);
|
|
|
|
auto rsa_from_pem = Crypto::PK::RSA(priv_pem);
|
|
|
|
|
2024-03-22 05:17:31 -04:00
|
|
|
EXPECT_EQ(keypem, StringView(priv_pem));
|
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
ByteBuffer msg_buffer = {};
|
|
|
|
msg_buffer.resize(rsa_from_pair.output_size());
|
2024-03-13 07:30:59 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
auto msg = msg_buffer.bytes();
|
|
|
|
msg.overwrite(0, "WellHelloFriends", 16);
|
2024-12-16 17:45:12 -05:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
auto enc = TRY_OR_FAIL(rsa_from_pair.encrypt(msg));
|
|
|
|
auto dec = TRY_OR_FAIL(rsa_from_pem.decrypt(enc));
|
2024-03-13 07:30:59 -04:00
|
|
|
|
|
|
|
EXPECT_EQ(memcmp(dec.data(), "WellHelloFriends", 16), 0);
|
|
|
|
}
|
|
|
|
|
2021-06-18 17:28:49 -04:00
|
|
|
TEST_CASE(test_RSA_encrypt_decrypt)
|
|
|
|
{
|
2024-12-23 11:41:37 -05:00
|
|
|
auto keypair = TRY_OR_FAIL(Crypto::PK::RSA::generate_key_pair(1024));
|
|
|
|
Crypto::PK::RSA rsa(keypair);
|
2021-06-18 17:28:49 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
ByteBuffer msg_buffer = {};
|
|
|
|
msg_buffer.resize(rsa.output_size());
|
2021-06-18 17:28:49 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
auto msg = msg_buffer.bytes();
|
|
|
|
msg.overwrite(0, "WellHelloFriendsWellHelloFriendsWellHelloFriendsWellHelloFriends", 64);
|
2021-06-18 17:28:49 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
auto enc = TRY_OR_FAIL(rsa.encrypt(msg));
|
|
|
|
auto dec = TRY_OR_FAIL(rsa.decrypt(enc));
|
2021-06-18 17:28:49 -04:00
|
|
|
|
2024-12-25 16:04:38 -05:00
|
|
|
EXPECT(memcmp(dec.data(), "WellHelloFriendsWellHelloFriendsWellHelloFriendsWellHelloFriends", 64) == 0);
|
2021-06-18 17:28:49 -04:00
|
|
|
}
|
2024-12-25 16:24:43 -05:00
|
|
|
|
|
|
|
TEST_CASE(test_RSA_sign_verify)
|
|
|
|
{
|
|
|
|
auto keypair = TRY_OR_FAIL(Crypto::PK::RSA::generate_key_pair(1024));
|
|
|
|
Crypto::PK::RSA rsa(keypair);
|
|
|
|
|
|
|
|
ByteBuffer msg_buffer = {};
|
|
|
|
msg_buffer.resize(rsa.output_size());
|
|
|
|
|
|
|
|
auto msg = msg_buffer.bytes();
|
|
|
|
msg.overwrite(0, "WellHelloFriendsWellHelloFriendsWellHelloFriendsWellHelloFriends", 64);
|
|
|
|
|
|
|
|
auto sig = TRY_OR_FAIL(rsa.sign(msg));
|
|
|
|
auto ok = TRY_OR_FAIL(rsa.verify(msg, sig));
|
|
|
|
EXPECT_EQ(ok, true);
|
|
|
|
}
|