react-native-quick-crypto 1.1.1 → 1.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/QuickCrypto.podspec +1 -0
- package/android/CMakeLists.txt +4 -0
- package/cpp/cipher/CCMCipher.cpp +7 -11
- package/cpp/cipher/ChaCha20Cipher.cpp +6 -10
- package/cpp/cipher/ChaCha20Poly1305Cipher.cpp +10 -16
- package/cpp/cipher/GCMCipher.cpp +3 -5
- package/cpp/cipher/HybridCipher.cpp +7 -13
- package/cpp/cipher/HybridRsaCipher.cpp +19 -27
- package/cpp/cipher/OCBCipher.cpp +2 -3
- package/cpp/cipher/XChaCha20Poly1305Cipher.cpp +13 -19
- package/cpp/cipher/XSalsa20Cipher.cpp +8 -12
- package/cpp/cipher/XSalsa20Poly1305Cipher.cpp +11 -16
- package/cpp/keys/HybridKeyObjectHandle.cpp +630 -2
- package/cpp/keys/HybridKeyObjectHandle.hpp +21 -1
- package/cpp/sign/HybridSignHandle.cpp +26 -8
- package/cpp/sign/HybridVerifyHandle.cpp +28 -11
- package/cpp/slhdsa/HybridSlhDsaKeyPair.cpp +245 -0
- package/cpp/slhdsa/HybridSlhDsaKeyPair.hpp +48 -0
- package/cpp/turboshake/HybridTurboShake.cpp +379 -0
- package/cpp/turboshake/HybridTurboShake.hpp +28 -0
- package/cpp/utils/HybridUtils.cpp +26 -14
- package/deps/blake3/README.md +6 -7
- package/deps/blake3/c/blake3.c +3 -2
- package/deps/blake3/c/blake3.h +2 -2
- package/deps/blake3/c/blake3_dispatch.c +2 -2
- package/deps/blake3/c/blake3_impl.h +1 -1
- package/deps/blake3/c/blake3_neon.c +5 -4
- package/deps/ncrypto/include/ncrypto/version.h +2 -2
- package/deps/ncrypto/include/ncrypto.h +9 -2
- package/deps/ncrypto/src/ncrypto.cpp +130 -35
- package/lib/commonjs/dhKeyPair.js +3 -0
- package/lib/commonjs/dhKeyPair.js.map +1 -1
- package/lib/commonjs/dsa.js +3 -0
- package/lib/commonjs/dsa.js.map +1 -1
- package/lib/commonjs/ec.js +37 -30
- package/lib/commonjs/ec.js.map +1 -1
- package/lib/commonjs/ed.js +60 -6
- package/lib/commonjs/ed.js.map +1 -1
- package/lib/commonjs/hash.js +52 -5
- package/lib/commonjs/hash.js.map +1 -1
- package/lib/commonjs/keys/classes.js +33 -7
- package/lib/commonjs/keys/classes.js.map +1 -1
- package/lib/commonjs/keys/generateKeyPair.js +85 -4
- package/lib/commonjs/keys/generateKeyPair.js.map +1 -1
- package/lib/commonjs/keys/index.js +50 -2
- package/lib/commonjs/keys/index.js.map +1 -1
- package/lib/commonjs/keys/signVerify.js +9 -2
- package/lib/commonjs/keys/signVerify.js.map +1 -1
- package/lib/commonjs/keys/utils.js +59 -1
- package/lib/commonjs/keys/utils.js.map +1 -1
- package/lib/commonjs/random.js +63 -9
- package/lib/commonjs/random.js.map +1 -1
- package/lib/commonjs/rsa.js +3 -0
- package/lib/commonjs/rsa.js.map +1 -1
- package/lib/commonjs/slhdsa.js +70 -0
- package/lib/commonjs/slhdsa.js.map +1 -0
- package/lib/commonjs/specs/slhDsaKeyPair.nitro.js +6 -0
- package/lib/commonjs/specs/slhDsaKeyPair.nitro.js.map +1 -0
- package/lib/commonjs/specs/turboshake.nitro.js +6 -0
- package/lib/commonjs/specs/turboshake.nitro.js.map +1 -0
- package/lib/commonjs/subtle.js +926 -275
- package/lib/commonjs/subtle.js.map +1 -1
- package/lib/commonjs/utils/conversion.js +53 -19
- package/lib/commonjs/utils/conversion.js.map +1 -1
- package/lib/commonjs/utils/errors.js +63 -4
- package/lib/commonjs/utils/errors.js.map +1 -1
- package/lib/commonjs/utils/types.js.map +1 -1
- package/lib/commonjs/utils/validation.js +46 -0
- package/lib/commonjs/utils/validation.js.map +1 -1
- package/lib/module/dhKeyPair.js +3 -0
- package/lib/module/dhKeyPair.js.map +1 -1
- package/lib/module/dsa.js +3 -0
- package/lib/module/dsa.js.map +1 -1
- package/lib/module/ec.js +38 -31
- package/lib/module/ec.js.map +1 -1
- package/lib/module/ed.js +61 -7
- package/lib/module/ed.js.map +1 -1
- package/lib/module/hash.js +52 -5
- package/lib/module/hash.js.map +1 -1
- package/lib/module/keys/classes.js +31 -5
- package/lib/module/keys/classes.js.map +1 -1
- package/lib/module/keys/generateKeyPair.js +86 -5
- package/lib/module/keys/generateKeyPair.js.map +1 -1
- package/lib/module/keys/index.js +50 -2
- package/lib/module/keys/index.js.map +1 -1
- package/lib/module/keys/signVerify.js +9 -2
- package/lib/module/keys/signVerify.js.map +1 -1
- package/lib/module/keys/utils.js +57 -1
- package/lib/module/keys/utils.js.map +1 -1
- package/lib/module/random.js +63 -10
- package/lib/module/random.js.map +1 -1
- package/lib/module/rsa.js +3 -0
- package/lib/module/rsa.js.map +1 -1
- package/lib/module/slhdsa.js +64 -0
- package/lib/module/slhdsa.js.map +1 -0
- package/lib/module/specs/slhDsaKeyPair.nitro.js +4 -0
- package/lib/module/specs/slhDsaKeyPair.nitro.js.map +1 -0
- package/lib/module/specs/turboshake.nitro.js +4 -0
- package/lib/module/specs/turboshake.nitro.js.map +1 -0
- package/lib/module/subtle.js +927 -276
- package/lib/module/subtle.js.map +1 -1
- package/lib/module/utils/conversion.js +51 -19
- package/lib/module/utils/conversion.js.map +1 -1
- package/lib/module/utils/errors.js +61 -4
- package/lib/module/utils/errors.js.map +1 -1
- package/lib/module/utils/types.js.map +1 -1
- package/lib/module/utils/validation.js +44 -0
- package/lib/module/utils/validation.js.map +1 -1
- package/lib/typescript/dhKeyPair.d.ts.map +1 -1
- package/lib/typescript/dsa.d.ts.map +1 -1
- package/lib/typescript/ec.d.ts.map +1 -1
- package/lib/typescript/ed.d.ts.map +1 -1
- package/lib/typescript/hash.d.ts.map +1 -1
- package/lib/typescript/index.d.ts +12 -7
- package/lib/typescript/index.d.ts.map +1 -1
- package/lib/typescript/keys/classes.d.ts +10 -1
- package/lib/typescript/keys/classes.d.ts.map +1 -1
- package/lib/typescript/keys/generateKeyPair.d.ts +12 -1
- package/lib/typescript/keys/generateKeyPair.d.ts.map +1 -1
- package/lib/typescript/keys/index.d.ts +3 -1
- package/lib/typescript/keys/index.d.ts.map +1 -1
- package/lib/typescript/keys/signVerify.d.ts.map +1 -1
- package/lib/typescript/keys/utils.d.ts +21 -4
- package/lib/typescript/keys/utils.d.ts.map +1 -1
- package/lib/typescript/random.d.ts +5 -1
- package/lib/typescript/random.d.ts.map +1 -1
- package/lib/typescript/rsa.d.ts.map +1 -1
- package/lib/typescript/slhdsa.d.ts +19 -0
- package/lib/typescript/slhdsa.d.ts.map +1 -0
- package/lib/typescript/specs/keyObjectHandle.nitro.d.ts +9 -0
- package/lib/typescript/specs/keyObjectHandle.nitro.d.ts.map +1 -1
- package/lib/typescript/specs/slhDsaKeyPair.nitro.d.ts +16 -0
- package/lib/typescript/specs/slhDsaKeyPair.nitro.d.ts.map +1 -0
- package/lib/typescript/specs/turboshake.nitro.d.ts +11 -0
- package/lib/typescript/specs/turboshake.nitro.d.ts.map +1 -0
- package/lib/typescript/subtle.d.ts +3 -2
- package/lib/typescript/subtle.d.ts.map +1 -1
- package/lib/typescript/utils/conversion.d.ts +4 -3
- package/lib/typescript/utils/conversion.d.ts.map +1 -1
- package/lib/typescript/utils/errors.d.ts +12 -0
- package/lib/typescript/utils/errors.d.ts.map +1 -1
- package/lib/typescript/utils/types.d.ts +32 -15
- package/lib/typescript/utils/types.d.ts.map +1 -1
- package/lib/typescript/utils/validation.d.ts +3 -1
- package/lib/typescript/utils/validation.d.ts.map +1 -1
- package/nitrogen/generated/android/QuickCrypto+autolinking.cmake +2 -0
- package/nitrogen/generated/android/QuickCryptoOnLoad.cpp +20 -0
- package/nitrogen/generated/ios/QuickCryptoAutolinking.mm +20 -0
- package/nitrogen/generated/shared/c++/AsymmetricKeyType.hpp +48 -0
- package/nitrogen/generated/shared/c++/HybridKeyObjectHandleSpec.cpp +9 -0
- package/nitrogen/generated/shared/c++/HybridKeyObjectHandleSpec.hpp +9 -0
- package/nitrogen/generated/shared/c++/HybridSlhDsaKeyPairSpec.cpp +29 -0
- package/nitrogen/generated/shared/c++/HybridSlhDsaKeyPairSpec.hpp +72 -0
- package/nitrogen/generated/shared/c++/HybridTurboShakeSpec.cpp +22 -0
- package/nitrogen/generated/shared/c++/HybridTurboShakeSpec.hpp +70 -0
- package/nitrogen/generated/shared/c++/JWK.hpp +9 -1
- package/nitrogen/generated/shared/c++/JWKkty.hpp +4 -0
- package/nitrogen/generated/shared/c++/KangarooTwelveVariant.hpp +76 -0
- package/nitrogen/generated/shared/c++/TurboShakeVariant.hpp +76 -0
- package/package.json +2 -3
- package/src/dhKeyPair.ts +8 -0
- package/src/dsa.ts +8 -0
- package/src/ec.ts +52 -29
- package/src/ed.ts +95 -16
- package/src/hash.ts +108 -5
- package/src/keys/classes.ts +46 -5
- package/src/keys/generateKeyPair.ts +151 -5
- package/src/keys/index.ts +73 -3
- package/src/keys/signVerify.ts +13 -2
- package/src/keys/utils.ts +78 -5
- package/src/random.ts +93 -9
- package/src/rsa.ts +8 -0
- package/src/slhdsa.ts +146 -0
- package/src/specs/keyObjectHandle.nitro.ts +17 -0
- package/src/specs/slhDsaKeyPair.nitro.ts +29 -0
- package/src/specs/turboshake.nitro.ts +21 -0
- package/src/subtle.ts +1191 -360
- package/src/utils/conversion.ts +72 -21
- package/src/utils/errors.ts +72 -4
- package/src/utils/types.ts +80 -15
- package/src/utils/validation.ts +70 -1
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
#include <cstdio>
|
|
2
|
+
#include <mutex>
|
|
2
3
|
#include <stdexcept>
|
|
3
4
|
|
|
4
5
|
#include "../utils/base64.h"
|
|
@@ -10,10 +11,42 @@
|
|
|
10
11
|
#include <openssl/ec.h>
|
|
11
12
|
#include <openssl/evp.h>
|
|
12
13
|
#include <openssl/obj_mac.h>
|
|
14
|
+
#include <openssl/provider.h>
|
|
13
15
|
#include <openssl/rsa.h>
|
|
14
16
|
|
|
15
17
|
namespace margelo::nitro::crypto {
|
|
16
18
|
|
|
19
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30600000L
|
|
20
|
+
// Configure loaded providers to prefer seed-only PKCS#8 output for ML-DSA /
|
|
21
|
+
// ML-KEM, falling back to priv-only when no seed is available. Without this,
|
|
22
|
+
// OpenSSL defaults to "seed-priv" — a longer encoding that bundles both —
|
|
23
|
+
// which breaks interop with Node and the exact-length export check in subtle.ts.
|
|
24
|
+
// Mirrors src/crypto/crypto_util.cc in Node.
|
|
25
|
+
static void configurePqcOutputFormats() {
|
|
26
|
+
static std::once_flag once;
|
|
27
|
+
std::call_once(once, []() {
|
|
28
|
+
OSSL_PROVIDER_do_all(
|
|
29
|
+
nullptr,
|
|
30
|
+
[](OSSL_PROVIDER* provider, void*) -> int {
|
|
31
|
+
OSSL_PROVIDER_add_conf_parameter(provider, "ml-kem.output_formats", "seed-only,priv-only");
|
|
32
|
+
OSSL_PROVIDER_add_conf_parameter(provider, "ml-dsa.output_formats", "seed-only,priv-only");
|
|
33
|
+
OSSL_PROVIDER_add_conf_parameter(provider, "slh-dsa.output_formats", "seed-only,priv-only");
|
|
34
|
+
return 1;
|
|
35
|
+
},
|
|
36
|
+
nullptr);
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
#endif
|
|
40
|
+
|
|
41
|
+
HybridKeyObjectHandle::HybridKeyObjectHandle() : HybridObject(TAG) {
|
|
42
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30600000L
|
|
43
|
+
// Configure once on first handle construction. Providers are guaranteed
|
|
44
|
+
// loaded by this point (any prior crypto op routed through ncrypto), and
|
|
45
|
+
// the call_once flag makes subsequent constructions cheap.
|
|
46
|
+
configurePqcOutputFormats();
|
|
47
|
+
#endif
|
|
48
|
+
}
|
|
49
|
+
|
|
17
50
|
// Helper functions for base64url encoding/decoding with BIGNUMs
|
|
18
51
|
static std::string bn_to_base64url(const BIGNUM* bn, size_t expected_size = 0) {
|
|
19
52
|
if (!bn)
|
|
@@ -142,7 +175,7 @@ std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportKey(std::optional<KFor
|
|
|
142
175
|
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
143
176
|
if (typeName != nullptr) {
|
|
144
177
|
std::string name(typeName);
|
|
145
|
-
bool isPqcKey = (name.starts_with("ML-KEM-") || name.starts_with("ML-DSA-"));
|
|
178
|
+
bool isPqcKey = (name.starts_with("ML-KEM-") || name.starts_with("ML-DSA-") || name.starts_with("SLH-DSA-"));
|
|
146
179
|
if (isPqcKey) {
|
|
147
180
|
if (keyType == KeyType::PUBLIC) {
|
|
148
181
|
auto rawData = pkey.rawPublicKey();
|
|
@@ -364,9 +397,267 @@ JWK HybridKeyObjectHandle::exportJwk(const JWK& key, bool handleRsaPss) {
|
|
|
364
397
|
return result;
|
|
365
398
|
}
|
|
366
399
|
|
|
400
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
401
|
+
// Export AKP keys (ML-DSA, ML-KEM)
|
|
402
|
+
{
|
|
403
|
+
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
404
|
+
if (typeName != nullptr) {
|
|
405
|
+
std::string name(typeName);
|
|
406
|
+
bool isPqcKey = (name.starts_with("ML-DSA-") || name.starts_with("ML-KEM-") || name.starts_with("SLH-DSA-"));
|
|
407
|
+
if (isPqcKey) {
|
|
408
|
+
result.kty = JWKkty::AKP;
|
|
409
|
+
result.alg = name;
|
|
410
|
+
|
|
411
|
+
auto pubKey = pkey.rawPublicKey();
|
|
412
|
+
if (!pubKey) {
|
|
413
|
+
throw std::runtime_error("Failed to get raw public key for AKP JWK export");
|
|
414
|
+
}
|
|
415
|
+
result.pub = base64url_encode(reinterpret_cast<const unsigned char*>(pubKey.get()), pubKey.size());
|
|
416
|
+
|
|
417
|
+
if (keyType == KeyType::PRIVATE) {
|
|
418
|
+
auto seed = pkey.rawSeed();
|
|
419
|
+
if (!seed) {
|
|
420
|
+
throw std::runtime_error("Key does not have an available seed");
|
|
421
|
+
}
|
|
422
|
+
result.priv = base64url_encode(reinterpret_cast<const unsigned char*>(seed.get()), seed.size());
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
return result;
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
#endif
|
|
430
|
+
|
|
367
431
|
throw std::runtime_error("Unsupported key type for JWK export");
|
|
368
432
|
}
|
|
369
433
|
|
|
434
|
+
// Returns true if the EVP_PKEY type supports raw public key export
|
|
435
|
+
// (CFRG keys: Ed25519, Ed448, X25519, X448; PQC keys: ML-DSA, ML-KEM, SLH-DSA).
|
|
436
|
+
static bool supportsRawPublic(int keyId, const char* typeName) {
|
|
437
|
+
if (keyId == EVP_PKEY_ED25519 || keyId == EVP_PKEY_ED448 || keyId == EVP_PKEY_X25519 || keyId == EVP_PKEY_X448) {
|
|
438
|
+
return true;
|
|
439
|
+
}
|
|
440
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
441
|
+
if (keyId == EVP_PKEY_ML_DSA_44 || keyId == EVP_PKEY_ML_DSA_65 || keyId == EVP_PKEY_ML_DSA_87) {
|
|
442
|
+
return true;
|
|
443
|
+
}
|
|
444
|
+
if (typeName != nullptr) {
|
|
445
|
+
std::string name(typeName);
|
|
446
|
+
if (name.starts_with("ML-KEM-") || name.starts_with("ML-DSA-") || name.starts_with("SLH-DSA-")) {
|
|
447
|
+
return true;
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
#else
|
|
451
|
+
(void)typeName;
|
|
452
|
+
#endif
|
|
453
|
+
return false;
|
|
454
|
+
}
|
|
455
|
+
|
|
456
|
+
// Returns true if the EVP_PKEY type supports raw private key export
|
|
457
|
+
// (CFRG keys: Ed25519, Ed448, X25519, X448; SLH-DSA private keys).
|
|
458
|
+
static bool supportsRawPrivate(int keyId, const char* typeName) {
|
|
459
|
+
if (keyId == EVP_PKEY_ED25519 || keyId == EVP_PKEY_ED448 || keyId == EVP_PKEY_X25519 || keyId == EVP_PKEY_X448) {
|
|
460
|
+
return true;
|
|
461
|
+
}
|
|
462
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
463
|
+
if (typeName != nullptr) {
|
|
464
|
+
std::string name(typeName);
|
|
465
|
+
if (name.starts_with("SLH-DSA-")) {
|
|
466
|
+
return true;
|
|
467
|
+
}
|
|
468
|
+
}
|
|
469
|
+
#else
|
|
470
|
+
(void)typeName;
|
|
471
|
+
#endif
|
|
472
|
+
return false;
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
// Returns true if the EVP_PKEY type supports raw seed export
|
|
476
|
+
// (PQC keys: ML-DSA, ML-KEM, SLH-DSA).
|
|
477
|
+
static bool supportsRawSeed(int keyId, const char* typeName) {
|
|
478
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
479
|
+
if (keyId == EVP_PKEY_ML_DSA_44 || keyId == EVP_PKEY_ML_DSA_65 || keyId == EVP_PKEY_ML_DSA_87) {
|
|
480
|
+
return true;
|
|
481
|
+
}
|
|
482
|
+
if (typeName != nullptr) {
|
|
483
|
+
std::string name(typeName);
|
|
484
|
+
if (name.starts_with("ML-KEM-") || name.starts_with("ML-DSA-") || name.starts_with("SLH-DSA-")) {
|
|
485
|
+
return true;
|
|
486
|
+
}
|
|
487
|
+
}
|
|
488
|
+
#else
|
|
489
|
+
(void)keyId;
|
|
490
|
+
(void)typeName;
|
|
491
|
+
#endif
|
|
492
|
+
return false;
|
|
493
|
+
}
|
|
494
|
+
|
|
495
|
+
std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportRawPublic() {
|
|
496
|
+
auto keyType = data_.GetKeyType();
|
|
497
|
+
if (keyType == KeyType::SECRET) {
|
|
498
|
+
throw std::runtime_error("Raw public key export is not supported for secret keys");
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
502
|
+
if (!pkey) {
|
|
503
|
+
throw std::runtime_error("Invalid asymmetric key");
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
int keyId = EVP_PKEY_id(pkey.get());
|
|
507
|
+
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
508
|
+
|
|
509
|
+
if (!supportsRawPublic(keyId, typeName)) {
|
|
510
|
+
throw std::runtime_error("The key type does not support raw public key export");
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
auto rawData = pkey.rawPublicKey();
|
|
514
|
+
if (!rawData) {
|
|
515
|
+
throw std::runtime_error("Failed to get raw public key");
|
|
516
|
+
}
|
|
517
|
+
return ToNativeArrayBuffer(reinterpret_cast<const uint8_t*>(rawData.get()), rawData.size());
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportRawPrivate() {
|
|
521
|
+
auto keyType = data_.GetKeyType();
|
|
522
|
+
if (keyType != KeyType::PRIVATE) {
|
|
523
|
+
throw std::runtime_error("Raw private key export requires a private key");
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
527
|
+
if (!pkey) {
|
|
528
|
+
throw std::runtime_error("Invalid asymmetric key");
|
|
529
|
+
}
|
|
530
|
+
|
|
531
|
+
int keyId = EVP_PKEY_id(pkey.get());
|
|
532
|
+
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
533
|
+
|
|
534
|
+
if (!supportsRawPrivate(keyId, typeName)) {
|
|
535
|
+
throw std::runtime_error("The key type does not support raw private key export");
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
auto rawData = pkey.rawPrivateKey();
|
|
539
|
+
if (!rawData) {
|
|
540
|
+
throw std::runtime_error("Failed to get raw private key");
|
|
541
|
+
}
|
|
542
|
+
return ToNativeArrayBuffer(reinterpret_cast<const uint8_t*>(rawData.get()), rawData.size());
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportRawSeed() {
|
|
546
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
547
|
+
auto keyType = data_.GetKeyType();
|
|
548
|
+
if (keyType != KeyType::PRIVATE) {
|
|
549
|
+
throw std::runtime_error("Raw seed export requires a private key");
|
|
550
|
+
}
|
|
551
|
+
|
|
552
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
553
|
+
if (!pkey) {
|
|
554
|
+
throw std::runtime_error("Invalid asymmetric key");
|
|
555
|
+
}
|
|
556
|
+
|
|
557
|
+
int keyId = EVP_PKEY_id(pkey.get());
|
|
558
|
+
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
559
|
+
|
|
560
|
+
if (!supportsRawSeed(keyId, typeName)) {
|
|
561
|
+
throw std::runtime_error("The key type does not support raw seed export");
|
|
562
|
+
}
|
|
563
|
+
|
|
564
|
+
auto rawData = pkey.rawSeed();
|
|
565
|
+
if (!rawData) {
|
|
566
|
+
throw std::runtime_error("Key does not have an available seed");
|
|
567
|
+
}
|
|
568
|
+
return ToNativeArrayBuffer(reinterpret_cast<const uint8_t*>(rawData.get()), rawData.size());
|
|
569
|
+
#else
|
|
570
|
+
throw std::runtime_error("Raw seed export requires OpenSSL 3.5+");
|
|
571
|
+
#endif
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportECPublicRaw(bool compressed) {
|
|
575
|
+
auto keyType = data_.GetKeyType();
|
|
576
|
+
if (keyType == KeyType::SECRET) {
|
|
577
|
+
throw std::runtime_error("EC raw public key export is not supported for secret keys");
|
|
578
|
+
}
|
|
579
|
+
|
|
580
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
581
|
+
if (!pkey) {
|
|
582
|
+
throw std::runtime_error("Invalid asymmetric key");
|
|
583
|
+
}
|
|
584
|
+
|
|
585
|
+
if (EVP_PKEY_id(pkey.get()) != EVP_PKEY_EC) {
|
|
586
|
+
throw std::runtime_error("Key is not an EC key");
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
const EC_KEY* ec_key = EVP_PKEY_get0_EC_KEY(pkey.get());
|
|
590
|
+
if (!ec_key) {
|
|
591
|
+
throw std::runtime_error("Failed to get EC key");
|
|
592
|
+
}
|
|
593
|
+
|
|
594
|
+
const EC_GROUP* group = EC_KEY_get0_group(ec_key);
|
|
595
|
+
const EC_POINT* point = EC_KEY_get0_public_key(ec_key);
|
|
596
|
+
if (!group || !point) {
|
|
597
|
+
throw std::runtime_error("Failed to get EC public key point");
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
point_conversion_form_t form = compressed ? POINT_CONVERSION_COMPRESSED : POINT_CONVERSION_UNCOMPRESSED;
|
|
601
|
+
|
|
602
|
+
size_t len = EC_POINT_point2oct(group, point, form, nullptr, 0, nullptr);
|
|
603
|
+
if (len == 0) {
|
|
604
|
+
throw std::runtime_error("Failed to compute EC point size");
|
|
605
|
+
}
|
|
606
|
+
std::vector<uint8_t> buf(len);
|
|
607
|
+
if (EC_POINT_point2oct(group, point, form, buf.data(), buf.size(), nullptr) != len) {
|
|
608
|
+
throw std::runtime_error("Failed to encode EC public key point");
|
|
609
|
+
}
|
|
610
|
+
return ToNativeArrayBuffer(buf.data(), buf.size());
|
|
611
|
+
}
|
|
612
|
+
|
|
613
|
+
std::shared_ptr<ArrayBuffer> HybridKeyObjectHandle::exportECPrivateRaw() {
|
|
614
|
+
auto keyType = data_.GetKeyType();
|
|
615
|
+
if (keyType != KeyType::PRIVATE) {
|
|
616
|
+
throw std::runtime_error("EC raw private key export requires a private key");
|
|
617
|
+
}
|
|
618
|
+
|
|
619
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
620
|
+
if (!pkey) {
|
|
621
|
+
throw std::runtime_error("Invalid asymmetric key");
|
|
622
|
+
}
|
|
623
|
+
|
|
624
|
+
if (EVP_PKEY_id(pkey.get()) != EVP_PKEY_EC) {
|
|
625
|
+
throw std::runtime_error("Key is not an EC key");
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
const EC_KEY* ec_key = EVP_PKEY_get0_EC_KEY(pkey.get());
|
|
629
|
+
if (!ec_key) {
|
|
630
|
+
throw std::runtime_error("Failed to get EC key");
|
|
631
|
+
}
|
|
632
|
+
|
|
633
|
+
const BIGNUM* priv_bn = EC_KEY_get0_private_key(ec_key);
|
|
634
|
+
if (!priv_bn) {
|
|
635
|
+
throw std::runtime_error("EC key has no private component");
|
|
636
|
+
}
|
|
637
|
+
|
|
638
|
+
const EC_GROUP* group = EC_KEY_get0_group(ec_key);
|
|
639
|
+
if (!group) {
|
|
640
|
+
throw std::runtime_error("Failed to get EC group");
|
|
641
|
+
}
|
|
642
|
+
|
|
643
|
+
BIGNUM* order = BN_new();
|
|
644
|
+
if (!order) {
|
|
645
|
+
throw std::runtime_error("Failed to allocate BIGNUM");
|
|
646
|
+
}
|
|
647
|
+
if (EC_GROUP_get_order(group, order, nullptr) != 1) {
|
|
648
|
+
BN_free(order);
|
|
649
|
+
throw std::runtime_error("Failed to get EC group order");
|
|
650
|
+
}
|
|
651
|
+
size_t order_size = (BN_num_bits(order) + 7) / 8;
|
|
652
|
+
BN_free(order);
|
|
653
|
+
|
|
654
|
+
std::vector<uint8_t> buf(order_size, 0);
|
|
655
|
+
if (BN_bn2binpad(priv_bn, buf.data(), static_cast<int>(order_size)) < 0) {
|
|
656
|
+
throw std::runtime_error("Failed to encode EC private key");
|
|
657
|
+
}
|
|
658
|
+
return ToNativeArrayBuffer(buf.data(), buf.size());
|
|
659
|
+
}
|
|
660
|
+
|
|
370
661
|
AsymmetricKeyType HybridKeyObjectHandle::getAsymmetricKeyType() {
|
|
371
662
|
const auto& pkey = data_.GetAsymmetricKey();
|
|
372
663
|
if (!pkey) {
|
|
@@ -407,7 +698,7 @@ AsymmetricKeyType HybridKeyObjectHandle::getAsymmetricKeyType() {
|
|
|
407
698
|
}
|
|
408
699
|
|
|
409
700
|
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
410
|
-
// EVP_PKEY_id returns -1 for provider-only key types (e.g. ML-KEM)
|
|
701
|
+
// EVP_PKEY_id returns -1 for provider-only key types (e.g. ML-KEM, SLH-DSA)
|
|
411
702
|
// Fall back to string-based type name comparison
|
|
412
703
|
const char* typeName = EVP_PKEY_get0_type_name(pkey.get());
|
|
413
704
|
if (typeName != nullptr) {
|
|
@@ -418,6 +709,30 @@ AsymmetricKeyType HybridKeyObjectHandle::getAsymmetricKeyType() {
|
|
|
418
709
|
return AsymmetricKeyType::ML_KEM_768;
|
|
419
710
|
if (name == "ML-KEM-1024")
|
|
420
711
|
return AsymmetricKeyType::ML_KEM_1024;
|
|
712
|
+
if (name == "SLH-DSA-SHA2-128s")
|
|
713
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_128S;
|
|
714
|
+
if (name == "SLH-DSA-SHA2-128f")
|
|
715
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_128F;
|
|
716
|
+
if (name == "SLH-DSA-SHA2-192s")
|
|
717
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_192S;
|
|
718
|
+
if (name == "SLH-DSA-SHA2-192f")
|
|
719
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_192F;
|
|
720
|
+
if (name == "SLH-DSA-SHA2-256s")
|
|
721
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_256S;
|
|
722
|
+
if (name == "SLH-DSA-SHA2-256f")
|
|
723
|
+
return AsymmetricKeyType::SLH_DSA_SHA2_256F;
|
|
724
|
+
if (name == "SLH-DSA-SHAKE-128s")
|
|
725
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_128S;
|
|
726
|
+
if (name == "SLH-DSA-SHAKE-128f")
|
|
727
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_128F;
|
|
728
|
+
if (name == "SLH-DSA-SHAKE-192s")
|
|
729
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_192S;
|
|
730
|
+
if (name == "SLH-DSA-SHAKE-192f")
|
|
731
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_192F;
|
|
732
|
+
if (name == "SLH-DSA-SHAKE-256s")
|
|
733
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_256S;
|
|
734
|
+
if (name == "SLH-DSA-SHAKE-256f")
|
|
735
|
+
return AsymmetricKeyType::SLH_DSA_SHAKE_256F;
|
|
421
736
|
}
|
|
422
737
|
#endif
|
|
423
738
|
|
|
@@ -703,6 +1018,96 @@ std::optional<KeyType> HybridKeyObjectHandle::initJwk(const JWK& keyData, std::o
|
|
|
703
1018
|
}
|
|
704
1019
|
}
|
|
705
1020
|
|
|
1021
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
1022
|
+
// Handle AKP keys (ML-DSA, ML-KEM)
|
|
1023
|
+
if (kty == JWKkty::AKP) {
|
|
1024
|
+
if (!keyData.alg.has_value()) {
|
|
1025
|
+
throw std::runtime_error("JWK AKP key missing 'alg' field");
|
|
1026
|
+
}
|
|
1027
|
+
if (!keyData.pub.has_value()) {
|
|
1028
|
+
throw std::runtime_error("JWK AKP key missing 'pub' field");
|
|
1029
|
+
}
|
|
1030
|
+
|
|
1031
|
+
const std::string& alg = keyData.alg.value();
|
|
1032
|
+
int nid = 0;
|
|
1033
|
+
if (alg == "ML-DSA-44")
|
|
1034
|
+
nid = EVP_PKEY_ML_DSA_44;
|
|
1035
|
+
else if (alg == "ML-DSA-65")
|
|
1036
|
+
nid = EVP_PKEY_ML_DSA_65;
|
|
1037
|
+
else if (alg == "ML-DSA-87")
|
|
1038
|
+
nid = EVP_PKEY_ML_DSA_87;
|
|
1039
|
+
else if (alg == "ML-KEM-512")
|
|
1040
|
+
nid = EVP_PKEY_ML_KEM_512;
|
|
1041
|
+
else if (alg == "ML-KEM-768")
|
|
1042
|
+
nid = EVP_PKEY_ML_KEM_768;
|
|
1043
|
+
else if (alg == "ML-KEM-1024")
|
|
1044
|
+
nid = EVP_PKEY_ML_KEM_1024;
|
|
1045
|
+
else if (alg == "SLH-DSA-SHA2-128s")
|
|
1046
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_128S;
|
|
1047
|
+
else if (alg == "SLH-DSA-SHA2-128f")
|
|
1048
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_128F;
|
|
1049
|
+
else if (alg == "SLH-DSA-SHA2-192s")
|
|
1050
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_192S;
|
|
1051
|
+
else if (alg == "SLH-DSA-SHA2-192f")
|
|
1052
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_192F;
|
|
1053
|
+
else if (alg == "SLH-DSA-SHA2-256s")
|
|
1054
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_256S;
|
|
1055
|
+
else if (alg == "SLH-DSA-SHA2-256f")
|
|
1056
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_256F;
|
|
1057
|
+
else if (alg == "SLH-DSA-SHAKE-128s")
|
|
1058
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_128S;
|
|
1059
|
+
else if (alg == "SLH-DSA-SHAKE-128f")
|
|
1060
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_128F;
|
|
1061
|
+
else if (alg == "SLH-DSA-SHAKE-192s")
|
|
1062
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_192S;
|
|
1063
|
+
else if (alg == "SLH-DSA-SHAKE-192f")
|
|
1064
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_192F;
|
|
1065
|
+
else if (alg == "SLH-DSA-SHAKE-256s")
|
|
1066
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_256S;
|
|
1067
|
+
else if (alg == "SLH-DSA-SHAKE-256f")
|
|
1068
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_256F;
|
|
1069
|
+
else
|
|
1070
|
+
throw std::runtime_error("Unsupported JWK AKP \"alg\": " + alg);
|
|
1071
|
+
|
|
1072
|
+
bool isPrivate = keyData.priv.has_value();
|
|
1073
|
+
ncrypto::EVPKeyPointer pkey;
|
|
1074
|
+
|
|
1075
|
+
if (isPrivate) {
|
|
1076
|
+
std::string seedBytes = base64url_decode(keyData.priv.value());
|
|
1077
|
+
ncrypto::Buffer<const unsigned char> buf{
|
|
1078
|
+
.data = reinterpret_cast<const unsigned char*>(seedBytes.data()),
|
|
1079
|
+
.len = seedBytes.size(),
|
|
1080
|
+
};
|
|
1081
|
+
pkey = ncrypto::EVPKeyPointer::NewRawSeed(nid, buf);
|
|
1082
|
+
if (!pkey) {
|
|
1083
|
+
throw std::runtime_error("Invalid JWK AKP key");
|
|
1084
|
+
}
|
|
1085
|
+
|
|
1086
|
+
// Verify the pub field matches the public key derived from the seed.
|
|
1087
|
+
std::string pubBytes = base64url_decode(keyData.pub.value());
|
|
1088
|
+
auto derivedPub = pkey.rawPublicKey();
|
|
1089
|
+
if (!derivedPub || derivedPub.size() != pubBytes.size() || CRYPTO_memcmp(derivedPub.get(), pubBytes.data(), pubBytes.size()) != 0) {
|
|
1090
|
+
throw std::runtime_error("Invalid JWK AKP key");
|
|
1091
|
+
}
|
|
1092
|
+
|
|
1093
|
+
data_ = KeyObjectData::CreateAsymmetric(KeyType::PRIVATE, std::move(pkey));
|
|
1094
|
+
return KeyType::PRIVATE;
|
|
1095
|
+
} else {
|
|
1096
|
+
std::string pubBytes = base64url_decode(keyData.pub.value());
|
|
1097
|
+
ncrypto::Buffer<const unsigned char> buf{
|
|
1098
|
+
.data = reinterpret_cast<const unsigned char*>(pubBytes.data()),
|
|
1099
|
+
.len = pubBytes.size(),
|
|
1100
|
+
};
|
|
1101
|
+
pkey = ncrypto::EVPKeyPointer::NewRawPublic(nid, buf);
|
|
1102
|
+
if (!pkey) {
|
|
1103
|
+
throw std::runtime_error("Invalid JWK AKP key");
|
|
1104
|
+
}
|
|
1105
|
+
data_ = KeyObjectData::CreateAsymmetric(KeyType::PUBLIC, std::move(pkey));
|
|
1106
|
+
return KeyType::PUBLIC;
|
|
1107
|
+
}
|
|
1108
|
+
}
|
|
1109
|
+
#endif
|
|
1110
|
+
|
|
706
1111
|
throw std::runtime_error("Unsupported JWK key type");
|
|
707
1112
|
}
|
|
708
1113
|
|
|
@@ -835,6 +1240,30 @@ bool HybridKeyObjectHandle::initPqcRaw(const std::string& algorithmName, const s
|
|
|
835
1240
|
nid = EVP_PKEY_ML_DSA_65;
|
|
836
1241
|
else if (algorithmName == "ML-DSA-87")
|
|
837
1242
|
nid = EVP_PKEY_ML_DSA_87;
|
|
1243
|
+
else if (algorithmName == "SLH-DSA-SHA2-128s")
|
|
1244
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_128S;
|
|
1245
|
+
else if (algorithmName == "SLH-DSA-SHA2-128f")
|
|
1246
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_128F;
|
|
1247
|
+
else if (algorithmName == "SLH-DSA-SHA2-192s")
|
|
1248
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_192S;
|
|
1249
|
+
else if (algorithmName == "SLH-DSA-SHA2-192f")
|
|
1250
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_192F;
|
|
1251
|
+
else if (algorithmName == "SLH-DSA-SHA2-256s")
|
|
1252
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_256S;
|
|
1253
|
+
else if (algorithmName == "SLH-DSA-SHA2-256f")
|
|
1254
|
+
nid = EVP_PKEY_SLH_DSA_SHA2_256F;
|
|
1255
|
+
else if (algorithmName == "SLH-DSA-SHAKE-128s")
|
|
1256
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_128S;
|
|
1257
|
+
else if (algorithmName == "SLH-DSA-SHAKE-128f")
|
|
1258
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_128F;
|
|
1259
|
+
else if (algorithmName == "SLH-DSA-SHAKE-192s")
|
|
1260
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_192S;
|
|
1261
|
+
else if (algorithmName == "SLH-DSA-SHAKE-192f")
|
|
1262
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_192F;
|
|
1263
|
+
else if (algorithmName == "SLH-DSA-SHAKE-256s")
|
|
1264
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_256S;
|
|
1265
|
+
else if (algorithmName == "SLH-DSA-SHAKE-256f")
|
|
1266
|
+
nid = EVP_PKEY_SLH_DSA_SHAKE_256F;
|
|
838
1267
|
else
|
|
839
1268
|
throw std::runtime_error("Unknown PQC algorithm: " + algorithmName);
|
|
840
1269
|
|
|
@@ -859,6 +1288,193 @@ bool HybridKeyObjectHandle::initPqcRaw(const std::string& algorithmName, const s
|
|
|
859
1288
|
#endif
|
|
860
1289
|
}
|
|
861
1290
|
|
|
1291
|
+
// Map a string asymmetricKeyType to an EVP_PKEY NID for OKP/PQC keys.
|
|
1292
|
+
// Returns 0 if the type is not a known OKP or PQC type.
|
|
1293
|
+
static int evpNidForAsymmetricKeyType(const std::string& asymmetricKeyType) {
|
|
1294
|
+
if (asymmetricKeyType == "ed25519")
|
|
1295
|
+
return EVP_PKEY_ED25519;
|
|
1296
|
+
if (asymmetricKeyType == "ed448")
|
|
1297
|
+
return EVP_PKEY_ED448;
|
|
1298
|
+
if (asymmetricKeyType == "x25519")
|
|
1299
|
+
return EVP_PKEY_X25519;
|
|
1300
|
+
if (asymmetricKeyType == "x448")
|
|
1301
|
+
return EVP_PKEY_X448;
|
|
1302
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
1303
|
+
if (asymmetricKeyType == "ml-dsa-44")
|
|
1304
|
+
return EVP_PKEY_ML_DSA_44;
|
|
1305
|
+
if (asymmetricKeyType == "ml-dsa-65")
|
|
1306
|
+
return EVP_PKEY_ML_DSA_65;
|
|
1307
|
+
if (asymmetricKeyType == "ml-dsa-87")
|
|
1308
|
+
return EVP_PKEY_ML_DSA_87;
|
|
1309
|
+
if (asymmetricKeyType == "ml-kem-512")
|
|
1310
|
+
return EVP_PKEY_ML_KEM_512;
|
|
1311
|
+
if (asymmetricKeyType == "ml-kem-768")
|
|
1312
|
+
return EVP_PKEY_ML_KEM_768;
|
|
1313
|
+
if (asymmetricKeyType == "ml-kem-1024")
|
|
1314
|
+
return EVP_PKEY_ML_KEM_1024;
|
|
1315
|
+
if (asymmetricKeyType == "slh-dsa-sha2-128s")
|
|
1316
|
+
return EVP_PKEY_SLH_DSA_SHA2_128S;
|
|
1317
|
+
if (asymmetricKeyType == "slh-dsa-sha2-128f")
|
|
1318
|
+
return EVP_PKEY_SLH_DSA_SHA2_128F;
|
|
1319
|
+
if (asymmetricKeyType == "slh-dsa-sha2-192s")
|
|
1320
|
+
return EVP_PKEY_SLH_DSA_SHA2_192S;
|
|
1321
|
+
if (asymmetricKeyType == "slh-dsa-sha2-192f")
|
|
1322
|
+
return EVP_PKEY_SLH_DSA_SHA2_192F;
|
|
1323
|
+
if (asymmetricKeyType == "slh-dsa-sha2-256s")
|
|
1324
|
+
return EVP_PKEY_SLH_DSA_SHA2_256S;
|
|
1325
|
+
if (asymmetricKeyType == "slh-dsa-sha2-256f")
|
|
1326
|
+
return EVP_PKEY_SLH_DSA_SHA2_256F;
|
|
1327
|
+
if (asymmetricKeyType == "slh-dsa-shake-128s")
|
|
1328
|
+
return EVP_PKEY_SLH_DSA_SHAKE_128S;
|
|
1329
|
+
if (asymmetricKeyType == "slh-dsa-shake-128f")
|
|
1330
|
+
return EVP_PKEY_SLH_DSA_SHAKE_128F;
|
|
1331
|
+
if (asymmetricKeyType == "slh-dsa-shake-192s")
|
|
1332
|
+
return EVP_PKEY_SLH_DSA_SHAKE_192S;
|
|
1333
|
+
if (asymmetricKeyType == "slh-dsa-shake-192f")
|
|
1334
|
+
return EVP_PKEY_SLH_DSA_SHAKE_192F;
|
|
1335
|
+
if (asymmetricKeyType == "slh-dsa-shake-256s")
|
|
1336
|
+
return EVP_PKEY_SLH_DSA_SHAKE_256S;
|
|
1337
|
+
if (asymmetricKeyType == "slh-dsa-shake-256f")
|
|
1338
|
+
return EVP_PKEY_SLH_DSA_SHAKE_256F;
|
|
1339
|
+
#endif
|
|
1340
|
+
return 0;
|
|
1341
|
+
}
|
|
1342
|
+
|
|
1343
|
+
bool HybridKeyObjectHandle::initRawPublic(const std::string& asymmetricKeyType, const std::shared_ptr<ArrayBuffer>& keyData,
|
|
1344
|
+
const std::optional<std::string>& namedCurve) {
|
|
1345
|
+
data_ = KeyObjectData();
|
|
1346
|
+
|
|
1347
|
+
if (asymmetricKeyType == "ec") {
|
|
1348
|
+
if (!namedCurve.has_value()) {
|
|
1349
|
+
throw std::runtime_error("namedCurve is required for EC raw public key import");
|
|
1350
|
+
}
|
|
1351
|
+
return initECRaw(namedCurve.value(), keyData);
|
|
1352
|
+
}
|
|
1353
|
+
|
|
1354
|
+
int nid = evpNidForAsymmetricKeyType(asymmetricKeyType);
|
|
1355
|
+
if (nid == 0) {
|
|
1356
|
+
throw std::runtime_error("Invalid asymmetricKeyType for raw public key import: " + asymmetricKeyType);
|
|
1357
|
+
}
|
|
1358
|
+
|
|
1359
|
+
ncrypto::Buffer<const unsigned char> buffer{.data = reinterpret_cast<const unsigned char*>(keyData->data()), .len = keyData->size()};
|
|
1360
|
+
auto pkey = ncrypto::EVPKeyPointer::NewRawPublic(nid, buffer);
|
|
1361
|
+
if (!pkey) {
|
|
1362
|
+
throw std::runtime_error("Failed to create raw public key");
|
|
1363
|
+
}
|
|
1364
|
+
this->data_ = KeyObjectData::CreateAsymmetric(KeyType::PUBLIC, std::move(pkey));
|
|
1365
|
+
return true;
|
|
1366
|
+
}
|
|
1367
|
+
|
|
1368
|
+
bool HybridKeyObjectHandle::initRawPrivate(const std::string& asymmetricKeyType, const std::shared_ptr<ArrayBuffer>& keyData,
|
|
1369
|
+
const std::optional<std::string>& namedCurve) {
|
|
1370
|
+
data_ = KeyObjectData();
|
|
1371
|
+
|
|
1372
|
+
if (asymmetricKeyType == "ec") {
|
|
1373
|
+
if (!namedCurve.has_value()) {
|
|
1374
|
+
throw std::runtime_error("namedCurve is required for EC raw private key import");
|
|
1375
|
+
}
|
|
1376
|
+
|
|
1377
|
+
int nid = 0;
|
|
1378
|
+
const std::string& curve = namedCurve.value();
|
|
1379
|
+
if (curve == "prime256v1" || curve == "P-256")
|
|
1380
|
+
nid = NID_X9_62_prime256v1;
|
|
1381
|
+
else if (curve == "secp384r1" || curve == "P-384")
|
|
1382
|
+
nid = NID_secp384r1;
|
|
1383
|
+
else if (curve == "secp521r1" || curve == "P-521")
|
|
1384
|
+
nid = NID_secp521r1;
|
|
1385
|
+
else if (curve == "secp256k1")
|
|
1386
|
+
nid = NID_secp256k1;
|
|
1387
|
+
else
|
|
1388
|
+
nid = OBJ_txt2nid(curve.c_str());
|
|
1389
|
+
|
|
1390
|
+
if (nid == 0) {
|
|
1391
|
+
throw std::runtime_error("Unknown curve: " + curve);
|
|
1392
|
+
}
|
|
1393
|
+
|
|
1394
|
+
auto ec_key = std::unique_ptr<EC_KEY, decltype(&EC_KEY_free)>(EC_KEY_new_by_curve_name(nid), EC_KEY_free);
|
|
1395
|
+
if (!ec_key) {
|
|
1396
|
+
throw std::runtime_error("Failed to create EC_KEY");
|
|
1397
|
+
}
|
|
1398
|
+
const EC_GROUP* group = EC_KEY_get0_group(ec_key.get());
|
|
1399
|
+
|
|
1400
|
+
BIGNUM* order = BN_new();
|
|
1401
|
+
if (!order || EC_GROUP_get_order(group, order, nullptr) != 1) {
|
|
1402
|
+
if (order)
|
|
1403
|
+
BN_free(order);
|
|
1404
|
+
throw std::runtime_error("Failed to get EC group order");
|
|
1405
|
+
}
|
|
1406
|
+
size_t order_size = (BN_num_bits(order) + 7) / 8;
|
|
1407
|
+
BN_free(order);
|
|
1408
|
+
|
|
1409
|
+
if (keyData->size() != order_size) {
|
|
1410
|
+
throw std::runtime_error("Invalid EC private key length");
|
|
1411
|
+
}
|
|
1412
|
+
|
|
1413
|
+
BIGNUM* priv_bn = BN_bin2bn(reinterpret_cast<const unsigned char*>(keyData->data()), static_cast<int>(keyData->size()), nullptr);
|
|
1414
|
+
if (!priv_bn) {
|
|
1415
|
+
throw std::runtime_error("Failed to decode EC private key");
|
|
1416
|
+
}
|
|
1417
|
+
|
|
1418
|
+
if (EC_KEY_set_private_key(ec_key.get(), priv_bn) != 1) {
|
|
1419
|
+
BN_free(priv_bn);
|
|
1420
|
+
throw std::runtime_error("Failed to set EC private key");
|
|
1421
|
+
}
|
|
1422
|
+
|
|
1423
|
+
auto pub_point = std::unique_ptr<EC_POINT, decltype(&EC_POINT_free)>(EC_POINT_new(group), EC_POINT_free);
|
|
1424
|
+
if (!pub_point || EC_POINT_mul(group, pub_point.get(), priv_bn, nullptr, nullptr, nullptr) != 1 ||
|
|
1425
|
+
EC_KEY_set_public_key(ec_key.get(), pub_point.get()) != 1) {
|
|
1426
|
+
BN_free(priv_bn);
|
|
1427
|
+
throw std::runtime_error("Failed to derive EC public key");
|
|
1428
|
+
}
|
|
1429
|
+
BN_free(priv_bn);
|
|
1430
|
+
|
|
1431
|
+
EVP_PKEY* pkey = EVP_PKEY_new();
|
|
1432
|
+
if (!pkey || EVP_PKEY_assign_EC_KEY(pkey, ec_key.get()) != 1) {
|
|
1433
|
+
if (pkey)
|
|
1434
|
+
EVP_PKEY_free(pkey);
|
|
1435
|
+
throw std::runtime_error("Failed to create EVP_PKEY from EC_KEY");
|
|
1436
|
+
}
|
|
1437
|
+
ec_key.release();
|
|
1438
|
+
|
|
1439
|
+
this->data_ = KeyObjectData::CreateAsymmetric(KeyType::PRIVATE, ncrypto::EVPKeyPointer(pkey));
|
|
1440
|
+
return true;
|
|
1441
|
+
}
|
|
1442
|
+
|
|
1443
|
+
int nid = evpNidForAsymmetricKeyType(asymmetricKeyType);
|
|
1444
|
+
if (nid == 0) {
|
|
1445
|
+
throw std::runtime_error("Invalid asymmetricKeyType for raw private key import: " + asymmetricKeyType);
|
|
1446
|
+
}
|
|
1447
|
+
|
|
1448
|
+
ncrypto::Buffer<const unsigned char> buffer{.data = reinterpret_cast<const unsigned char*>(keyData->data()), .len = keyData->size()};
|
|
1449
|
+
auto pkey = ncrypto::EVPKeyPointer::NewRawPrivate(nid, buffer);
|
|
1450
|
+
if (!pkey) {
|
|
1451
|
+
throw std::runtime_error("Failed to create raw private key");
|
|
1452
|
+
}
|
|
1453
|
+
this->data_ = KeyObjectData::CreateAsymmetric(KeyType::PRIVATE, std::move(pkey));
|
|
1454
|
+
return true;
|
|
1455
|
+
}
|
|
1456
|
+
|
|
1457
|
+
bool HybridKeyObjectHandle::initRawSeed(const std::string& asymmetricKeyType, const std::shared_ptr<ArrayBuffer>& keyData) {
|
|
1458
|
+
#if OPENSSL_VERSION_NUMBER >= 0x30500000L
|
|
1459
|
+
data_ = KeyObjectData();
|
|
1460
|
+
|
|
1461
|
+
int nid = evpNidForAsymmetricKeyType(asymmetricKeyType);
|
|
1462
|
+
if (nid == 0) {
|
|
1463
|
+
throw std::runtime_error("Invalid asymmetricKeyType for raw seed import: " + asymmetricKeyType);
|
|
1464
|
+
}
|
|
1465
|
+
|
|
1466
|
+
ncrypto::Buffer<const unsigned char> buffer{.data = reinterpret_cast<const unsigned char*>(keyData->data()), .len = keyData->size()};
|
|
1467
|
+
auto pkey = ncrypto::EVPKeyPointer::NewRawSeed(nid, buffer);
|
|
1468
|
+
if (!pkey) {
|
|
1469
|
+
throw std::runtime_error("Failed to create key from raw seed");
|
|
1470
|
+
}
|
|
1471
|
+
this->data_ = KeyObjectData::CreateAsymmetric(KeyType::PRIVATE, std::move(pkey));
|
|
1472
|
+
return true;
|
|
1473
|
+
#else
|
|
1474
|
+
throw std::runtime_error("Raw seed import requires OpenSSL 3.5+");
|
|
1475
|
+
#endif
|
|
1476
|
+
}
|
|
1477
|
+
|
|
862
1478
|
bool HybridKeyObjectHandle::keyEquals(const std::shared_ptr<HybridKeyObjectHandleSpec>& other) {
|
|
863
1479
|
auto otherHandle = std::dynamic_pointer_cast<HybridKeyObjectHandle>(other);
|
|
864
1480
|
if (!otherHandle)
|
|
@@ -887,4 +1503,16 @@ double HybridKeyObjectHandle::getSymmetricKeySize() {
|
|
|
887
1503
|
return static_cast<double>(data_.GetSymmetricKeySize());
|
|
888
1504
|
}
|
|
889
1505
|
|
|
1506
|
+
bool HybridKeyObjectHandle::checkEcKeyData() {
|
|
1507
|
+
const auto& pkey = data_.GetAsymmetricKey();
|
|
1508
|
+
if (!pkey || EVP_PKEY_id(pkey.get()) != EVP_PKEY_EC) {
|
|
1509
|
+
return false;
|
|
1510
|
+
}
|
|
1511
|
+
auto ctx = pkey.newCtx();
|
|
1512
|
+
if (!ctx) {
|
|
1513
|
+
return false;
|
|
1514
|
+
}
|
|
1515
|
+
return data_.GetKeyType() == KeyType::PRIVATE ? ctx.privateCheck() : ctx.publicCheck();
|
|
1516
|
+
}
|
|
1517
|
+
|
|
890
1518
|
} // namespace margelo::nitro::crypto
|