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.
Files changed (181) hide show
  1. package/QuickCrypto.podspec +1 -0
  2. package/android/CMakeLists.txt +4 -0
  3. package/cpp/cipher/CCMCipher.cpp +7 -11
  4. package/cpp/cipher/ChaCha20Cipher.cpp +6 -10
  5. package/cpp/cipher/ChaCha20Poly1305Cipher.cpp +10 -16
  6. package/cpp/cipher/GCMCipher.cpp +3 -5
  7. package/cpp/cipher/HybridCipher.cpp +7 -13
  8. package/cpp/cipher/HybridRsaCipher.cpp +19 -27
  9. package/cpp/cipher/OCBCipher.cpp +2 -3
  10. package/cpp/cipher/XChaCha20Poly1305Cipher.cpp +13 -19
  11. package/cpp/cipher/XSalsa20Cipher.cpp +8 -12
  12. package/cpp/cipher/XSalsa20Poly1305Cipher.cpp +11 -16
  13. package/cpp/keys/HybridKeyObjectHandle.cpp +630 -2
  14. package/cpp/keys/HybridKeyObjectHandle.hpp +21 -1
  15. package/cpp/sign/HybridSignHandle.cpp +26 -8
  16. package/cpp/sign/HybridVerifyHandle.cpp +28 -11
  17. package/cpp/slhdsa/HybridSlhDsaKeyPair.cpp +245 -0
  18. package/cpp/slhdsa/HybridSlhDsaKeyPair.hpp +48 -0
  19. package/cpp/turboshake/HybridTurboShake.cpp +379 -0
  20. package/cpp/turboshake/HybridTurboShake.hpp +28 -0
  21. package/cpp/utils/HybridUtils.cpp +26 -14
  22. package/deps/blake3/README.md +6 -7
  23. package/deps/blake3/c/blake3.c +3 -2
  24. package/deps/blake3/c/blake3.h +2 -2
  25. package/deps/blake3/c/blake3_dispatch.c +2 -2
  26. package/deps/blake3/c/blake3_impl.h +1 -1
  27. package/deps/blake3/c/blake3_neon.c +5 -4
  28. package/deps/ncrypto/include/ncrypto/version.h +2 -2
  29. package/deps/ncrypto/include/ncrypto.h +9 -2
  30. package/deps/ncrypto/src/ncrypto.cpp +130 -35
  31. package/lib/commonjs/dhKeyPair.js +3 -0
  32. package/lib/commonjs/dhKeyPair.js.map +1 -1
  33. package/lib/commonjs/dsa.js +3 -0
  34. package/lib/commonjs/dsa.js.map +1 -1
  35. package/lib/commonjs/ec.js +37 -30
  36. package/lib/commonjs/ec.js.map +1 -1
  37. package/lib/commonjs/ed.js +60 -6
  38. package/lib/commonjs/ed.js.map +1 -1
  39. package/lib/commonjs/hash.js +52 -5
  40. package/lib/commonjs/hash.js.map +1 -1
  41. package/lib/commonjs/keys/classes.js +33 -7
  42. package/lib/commonjs/keys/classes.js.map +1 -1
  43. package/lib/commonjs/keys/generateKeyPair.js +85 -4
  44. package/lib/commonjs/keys/generateKeyPair.js.map +1 -1
  45. package/lib/commonjs/keys/index.js +50 -2
  46. package/lib/commonjs/keys/index.js.map +1 -1
  47. package/lib/commonjs/keys/signVerify.js +9 -2
  48. package/lib/commonjs/keys/signVerify.js.map +1 -1
  49. package/lib/commonjs/keys/utils.js +59 -1
  50. package/lib/commonjs/keys/utils.js.map +1 -1
  51. package/lib/commonjs/random.js +63 -9
  52. package/lib/commonjs/random.js.map +1 -1
  53. package/lib/commonjs/rsa.js +3 -0
  54. package/lib/commonjs/rsa.js.map +1 -1
  55. package/lib/commonjs/slhdsa.js +70 -0
  56. package/lib/commonjs/slhdsa.js.map +1 -0
  57. package/lib/commonjs/specs/slhDsaKeyPair.nitro.js +6 -0
  58. package/lib/commonjs/specs/slhDsaKeyPair.nitro.js.map +1 -0
  59. package/lib/commonjs/specs/turboshake.nitro.js +6 -0
  60. package/lib/commonjs/specs/turboshake.nitro.js.map +1 -0
  61. package/lib/commonjs/subtle.js +926 -275
  62. package/lib/commonjs/subtle.js.map +1 -1
  63. package/lib/commonjs/utils/conversion.js +53 -19
  64. package/lib/commonjs/utils/conversion.js.map +1 -1
  65. package/lib/commonjs/utils/errors.js +63 -4
  66. package/lib/commonjs/utils/errors.js.map +1 -1
  67. package/lib/commonjs/utils/types.js.map +1 -1
  68. package/lib/commonjs/utils/validation.js +46 -0
  69. package/lib/commonjs/utils/validation.js.map +1 -1
  70. package/lib/module/dhKeyPair.js +3 -0
  71. package/lib/module/dhKeyPair.js.map +1 -1
  72. package/lib/module/dsa.js +3 -0
  73. package/lib/module/dsa.js.map +1 -1
  74. package/lib/module/ec.js +38 -31
  75. package/lib/module/ec.js.map +1 -1
  76. package/lib/module/ed.js +61 -7
  77. package/lib/module/ed.js.map +1 -1
  78. package/lib/module/hash.js +52 -5
  79. package/lib/module/hash.js.map +1 -1
  80. package/lib/module/keys/classes.js +31 -5
  81. package/lib/module/keys/classes.js.map +1 -1
  82. package/lib/module/keys/generateKeyPair.js +86 -5
  83. package/lib/module/keys/generateKeyPair.js.map +1 -1
  84. package/lib/module/keys/index.js +50 -2
  85. package/lib/module/keys/index.js.map +1 -1
  86. package/lib/module/keys/signVerify.js +9 -2
  87. package/lib/module/keys/signVerify.js.map +1 -1
  88. package/lib/module/keys/utils.js +57 -1
  89. package/lib/module/keys/utils.js.map +1 -1
  90. package/lib/module/random.js +63 -10
  91. package/lib/module/random.js.map +1 -1
  92. package/lib/module/rsa.js +3 -0
  93. package/lib/module/rsa.js.map +1 -1
  94. package/lib/module/slhdsa.js +64 -0
  95. package/lib/module/slhdsa.js.map +1 -0
  96. package/lib/module/specs/slhDsaKeyPair.nitro.js +4 -0
  97. package/lib/module/specs/slhDsaKeyPair.nitro.js.map +1 -0
  98. package/lib/module/specs/turboshake.nitro.js +4 -0
  99. package/lib/module/specs/turboshake.nitro.js.map +1 -0
  100. package/lib/module/subtle.js +927 -276
  101. package/lib/module/subtle.js.map +1 -1
  102. package/lib/module/utils/conversion.js +51 -19
  103. package/lib/module/utils/conversion.js.map +1 -1
  104. package/lib/module/utils/errors.js +61 -4
  105. package/lib/module/utils/errors.js.map +1 -1
  106. package/lib/module/utils/types.js.map +1 -1
  107. package/lib/module/utils/validation.js +44 -0
  108. package/lib/module/utils/validation.js.map +1 -1
  109. package/lib/typescript/dhKeyPair.d.ts.map +1 -1
  110. package/lib/typescript/dsa.d.ts.map +1 -1
  111. package/lib/typescript/ec.d.ts.map +1 -1
  112. package/lib/typescript/ed.d.ts.map +1 -1
  113. package/lib/typescript/hash.d.ts.map +1 -1
  114. package/lib/typescript/index.d.ts +12 -7
  115. package/lib/typescript/index.d.ts.map +1 -1
  116. package/lib/typescript/keys/classes.d.ts +10 -1
  117. package/lib/typescript/keys/classes.d.ts.map +1 -1
  118. package/lib/typescript/keys/generateKeyPair.d.ts +12 -1
  119. package/lib/typescript/keys/generateKeyPair.d.ts.map +1 -1
  120. package/lib/typescript/keys/index.d.ts +3 -1
  121. package/lib/typescript/keys/index.d.ts.map +1 -1
  122. package/lib/typescript/keys/signVerify.d.ts.map +1 -1
  123. package/lib/typescript/keys/utils.d.ts +21 -4
  124. package/lib/typescript/keys/utils.d.ts.map +1 -1
  125. package/lib/typescript/random.d.ts +5 -1
  126. package/lib/typescript/random.d.ts.map +1 -1
  127. package/lib/typescript/rsa.d.ts.map +1 -1
  128. package/lib/typescript/slhdsa.d.ts +19 -0
  129. package/lib/typescript/slhdsa.d.ts.map +1 -0
  130. package/lib/typescript/specs/keyObjectHandle.nitro.d.ts +9 -0
  131. package/lib/typescript/specs/keyObjectHandle.nitro.d.ts.map +1 -1
  132. package/lib/typescript/specs/slhDsaKeyPair.nitro.d.ts +16 -0
  133. package/lib/typescript/specs/slhDsaKeyPair.nitro.d.ts.map +1 -0
  134. package/lib/typescript/specs/turboshake.nitro.d.ts +11 -0
  135. package/lib/typescript/specs/turboshake.nitro.d.ts.map +1 -0
  136. package/lib/typescript/subtle.d.ts +3 -2
  137. package/lib/typescript/subtle.d.ts.map +1 -1
  138. package/lib/typescript/utils/conversion.d.ts +4 -3
  139. package/lib/typescript/utils/conversion.d.ts.map +1 -1
  140. package/lib/typescript/utils/errors.d.ts +12 -0
  141. package/lib/typescript/utils/errors.d.ts.map +1 -1
  142. package/lib/typescript/utils/types.d.ts +32 -15
  143. package/lib/typescript/utils/types.d.ts.map +1 -1
  144. package/lib/typescript/utils/validation.d.ts +3 -1
  145. package/lib/typescript/utils/validation.d.ts.map +1 -1
  146. package/nitrogen/generated/android/QuickCrypto+autolinking.cmake +2 -0
  147. package/nitrogen/generated/android/QuickCryptoOnLoad.cpp +20 -0
  148. package/nitrogen/generated/ios/QuickCryptoAutolinking.mm +20 -0
  149. package/nitrogen/generated/shared/c++/AsymmetricKeyType.hpp +48 -0
  150. package/nitrogen/generated/shared/c++/HybridKeyObjectHandleSpec.cpp +9 -0
  151. package/nitrogen/generated/shared/c++/HybridKeyObjectHandleSpec.hpp +9 -0
  152. package/nitrogen/generated/shared/c++/HybridSlhDsaKeyPairSpec.cpp +29 -0
  153. package/nitrogen/generated/shared/c++/HybridSlhDsaKeyPairSpec.hpp +72 -0
  154. package/nitrogen/generated/shared/c++/HybridTurboShakeSpec.cpp +22 -0
  155. package/nitrogen/generated/shared/c++/HybridTurboShakeSpec.hpp +70 -0
  156. package/nitrogen/generated/shared/c++/JWK.hpp +9 -1
  157. package/nitrogen/generated/shared/c++/JWKkty.hpp +4 -0
  158. package/nitrogen/generated/shared/c++/KangarooTwelveVariant.hpp +76 -0
  159. package/nitrogen/generated/shared/c++/TurboShakeVariant.hpp +76 -0
  160. package/package.json +2 -3
  161. package/src/dhKeyPair.ts +8 -0
  162. package/src/dsa.ts +8 -0
  163. package/src/ec.ts +52 -29
  164. package/src/ed.ts +95 -16
  165. package/src/hash.ts +108 -5
  166. package/src/keys/classes.ts +46 -5
  167. package/src/keys/generateKeyPair.ts +151 -5
  168. package/src/keys/index.ts +73 -3
  169. package/src/keys/signVerify.ts +13 -2
  170. package/src/keys/utils.ts +78 -5
  171. package/src/random.ts +93 -9
  172. package/src/rsa.ts +8 -0
  173. package/src/slhdsa.ts +146 -0
  174. package/src/specs/keyObjectHandle.nitro.ts +17 -0
  175. package/src/specs/slhDsaKeyPair.nitro.ts +29 -0
  176. package/src/specs/turboshake.nitro.ts +21 -0
  177. package/src/subtle.ts +1191 -360
  178. package/src/utils/conversion.ts +72 -21
  179. package/src/utils/errors.ts +72 -4
  180. package/src/utils/types.ts +80 -15
  181. 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