Skip to content

Commit 3888bab

Browse files
committed
feat: add Jwk::from_decoding_key
Allow for constructing a Jwk from a decoding key. This allows it to be created from a DER encoded file, for example. This patch renames JwkUtils to KeyUtils.
1 parent a89c743 commit 3888bab

4 files changed

Lines changed: 192 additions & 47 deletions

File tree

src/crypto/aws_lc/mod.rs

Lines changed: 16 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use aws_lc_rs::{
88

99
use crate::{
1010
Algorithm, DecodingKey, EncodingKey,
11-
crypto::{CryptoProvider, JwkUtils, JwtSigner, JwtVerifier},
11+
crypto::{CryptoProvider, JwtSigner, JwtVerifier, KeyUtils},
1212
errors::{self, Error, ErrorKind},
1313
jwk::{EllipticCurve, ThumbprintHash},
1414
};
@@ -18,15 +18,23 @@ mod eddsa;
1818
mod hmac;
1919
mod rsa;
2020

21-
fn extract_rsa_public_key_components(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
21+
fn rsa_components_from_private_key(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
2222
let key_pair = aws_sig::RsaKeyPair::from_der(key_content)
2323
.map_err(|e| ErrorKind::InvalidRsaKey(e.to_string()))?;
2424
let public = key_pair.public_key();
2525
let components = aws_sig::RsaPublicKeyComponents::<Vec<u8>>::from(public);
2626
Ok((components.n, components.e))
2727
}
2828

29-
fn extract_ec_public_key_coordinates(
29+
fn rsa_components_from_public_key(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
30+
let public = aws_lc_rs::rsa::PublicKey::from_der(key_content)
31+
.map_err(|e| ErrorKind::InvalidRsaKey(e.to_string()))?;
32+
33+
let components = aws_sig::RsaPublicKeyComponents::<Vec<u8>>::from(&public);
34+
Ok((components.n, components.e))
35+
}
36+
37+
fn ec_components_from_private_key(
3038
key_content: &[u8],
3139
alg: Algorithm,
3240
) -> errors::Result<(EllipticCurve, Vec<u8>, Vec<u8>)> {
@@ -102,9 +110,11 @@ fn new_verifier(
102110
pub static DEFAULT_PROVIDER: CryptoProvider = CryptoProvider {
103111
signer_factory: new_signer,
104112
verifier_factory: new_verifier,
105-
jwk_utils: JwkUtils {
106-
extract_rsa_public_key_components,
107-
extract_ec_public_key_coordinates,
113+
key_utils: KeyUtils {
114+
rsa_pub_components_from_private_key: rsa_components_from_private_key,
115+
rsa_pub_components_from_public_key: rsa_components_from_public_key,
116+
ec_pub_components_from_private_key: ec_components_from_private_key,
117+
ec_pub_components_from_public_key: crate::crypto::ec_components_from_public_key,
108118
compute_digest,
109119
},
110120
};

src/crypto/mod.rs

Lines changed: 41 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
//! [`CryptoProvider`]: crate::crypto::CryptoProvider
1111
1212
use crate::algorithms::Algorithm;
13-
use crate::errors::Result;
13+
use crate::errors::{self, ErrorKind, Result};
1414
use crate::jwk::{EllipticCurve, ThumbprintHash};
1515
use crate::{DecodingKey, EncodingKey};
1616

@@ -86,7 +86,7 @@ pub struct CryptoProvider {
8686
/// A function that produces a [`JwtVerifier`] for a given [`Algorithm`]
8787
pub verifier_factory: fn(&Algorithm, &DecodingKey) -> Result<Box<dyn JwtVerifier>>,
8888
/// Struct with utility functions for JWK processing.
89-
pub jwk_utils: JwkUtils,
89+
pub key_utils: KeyUtils,
9090
}
9191

9292
impl CryptoProvider {
@@ -123,7 +123,7 @@ See the documentation of the CryptoProvider type for more information.
123123
static INSTANCE: CryptoProvider = CryptoProvider {
124124
signer_factory: |_, _| panic!("{}", NOT_INSTALLED_ERROR),
125125
verifier_factory: |_, _| panic!("{}", NOT_INSTALLED_ERROR),
126-
jwk_utils: JwkUtils::new_unimplemented(),
126+
key_utils: KeyUtils::new_unimplemented(),
127127
};
128128

129129
&INSTANCE
@@ -132,22 +132,29 @@ See the documentation of the CryptoProvider type for more information.
132132
}
133133

134134
/// Holds utility functions required for JWK processing.
135-
/// Use the [`JwkUtils::new_unimplemented`] function to initialize all values to dummies.
135+
/// Use the [`KeyUtils::new_unimplemented`] function to initialize all values to dummies.
136136
#[derive(Clone, Debug)]
137-
pub struct JwkUtils {
137+
pub struct KeyUtils {
138138
/// Given a DER encoded private key, extract the RSA public key components (n, e)
139139
#[allow(clippy::type_complexity)]
140-
pub extract_rsa_public_key_components: fn(&[u8]) -> Result<(Vec<u8>, Vec<u8>)>,
140+
pub rsa_pub_components_from_private_key: fn(&[u8]) -> Result<(Vec<u8>, Vec<u8>)>,
141+
/// Given a DER encoded public key, extract the RSA public key components (n, e)
142+
#[allow(clippy::type_complexity)]
143+
pub rsa_pub_components_from_public_key: fn(&[u8]) -> Result<(Vec<u8>, Vec<u8>)>,
141144
/// Given a DER encoded private key and an algorithm, extract the associated curve
142145
/// and the EC public key components (x, y)
143146
#[allow(clippy::type_complexity)]
144-
pub extract_ec_public_key_coordinates:
147+
pub ec_pub_components_from_private_key:
145148
fn(&[u8], Algorithm) -> Result<(EllipticCurve, Vec<u8>, Vec<u8>)>,
149+
/// Given bitstring from DER encoded private key, extract the associated curve
150+
/// and the EC public key components (x, y)
151+
#[allow(clippy::type_complexity)]
152+
pub ec_pub_components_from_public_key: fn(&[u8]) -> Result<(EllipticCurve, Vec<u8>, Vec<u8>)>,
146153
/// Given some data and a name of a hash function, compute hash_function(data)
147154
pub compute_digest: fn(&[u8], ThumbprintHash) -> Result<Vec<u8>>,
148155
}
149156

150-
impl JwkUtils {
157+
impl KeyUtils {
151158
/// Initialises all values to dummies.
152159
/// Will lead to a panic when JWKs are required, so only use it if you don't want to support JWKs.
153160
pub const fn new_unimplemented() -> Self {
@@ -157,17 +164,41 @@ Call CryptoProvider::install_default() before this point to select a provider ma
157164
See the documentation of the CryptoProvider type for more information.
158165
"###;
159166
Self {
160-
extract_rsa_public_key_components: |_| {
167+
rsa_pub_components_from_private_key: |_| {
161168
panic!("{}", NOT_INSTALLED_OR_UNIMPLEMENTED_ERROR)
162169
},
163-
extract_ec_public_key_coordinates: |_, _| {
170+
rsa_pub_components_from_public_key: |_| {
171+
panic!("{}", NOT_INSTALLED_OR_UNIMPLEMENTED_ERROR)
172+
},
173+
ec_pub_components_from_private_key: |_, _| {
174+
panic!("{}", NOT_INSTALLED_OR_UNIMPLEMENTED_ERROR)
175+
},
176+
ec_pub_components_from_public_key: |_| {
164177
panic!("{}", NOT_INSTALLED_OR_UNIMPLEMENTED_ERROR)
165178
},
166179
compute_digest: |_, _| panic!("{}", NOT_INSTALLED_OR_UNIMPLEMENTED_ERROR),
167180
}
168181
}
169182
}
170183

184+
#[allow(unused)]
185+
fn ec_components_from_public_key(
186+
pub_bytes: &[u8],
187+
) -> errors::Result<(EllipticCurve, Vec<u8>, Vec<u8>)> {
188+
let (curve, pub_elem_bytes) = match pub_bytes.len() {
189+
65 => (EllipticCurve::P256, 32),
190+
97 => (EllipticCurve::P384, 48),
191+
_ => return Err(ErrorKind::InvalidEcdsaKey.into()),
192+
};
193+
194+
if pub_bytes[0] != 4 {
195+
return Err(ErrorKind::InvalidEcdsaKey.into());
196+
}
197+
198+
let (x, y) = pub_bytes[1..].split_at(pub_elem_bytes);
199+
Ok((curve, x.to_vec(), y.to_vec()))
200+
}
201+
171202
mod static_default {
172203
use std::sync::OnceLock;
173204

src/crypto/rust_crypto/mod.rs

Lines changed: 19 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,15 @@
1-
use ::rsa::{RsaPrivateKey, pkcs1::DecodeRsaPrivateKey, traits::PublicKeyParts};
1+
use ::rsa::{
2+
RsaPrivateKey, RsaPublicKey,
3+
pkcs1::{DecodeRsaPrivateKey, DecodeRsaPublicKey},
4+
traits::PublicKeyParts,
5+
};
26
use p256::{ecdsa::SigningKey as P256SigningKey, pkcs8::DecodePrivateKey};
37
use p384::ecdsa::SigningKey as P384SigningKey;
48
use sha2::{Digest, Sha256, Sha384, Sha512};
59

610
use crate::{
711
Algorithm, DecodingKey, EncodingKey,
8-
crypto::{CryptoProvider, JwkUtils, JwtSigner, JwtVerifier},
12+
crypto::{CryptoProvider, JwtSigner, JwtVerifier, KeyUtils},
913
errors::{self, Error, ErrorKind},
1014
jwk::{EllipticCurve, ThumbprintHash},
1115
};
@@ -15,14 +19,20 @@ mod eddsa;
1519
mod hmac;
1620
mod rsa;
1721

18-
fn extract_rsa_public_key_components(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
22+
fn rsa_components_from_private_key(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
1923
let private_key = RsaPrivateKey::from_pkcs1_der(key_content)
2024
.map_err(|e| ErrorKind::InvalidRsaKey(e.to_string()))?;
2125
let public_key = private_key.to_public_key();
2226
Ok((public_key.n().to_bytes_be(), public_key.e().to_bytes_be()))
2327
}
2428

25-
fn extract_ec_public_key_coordinates(
29+
fn rsa_components_from_public_key(key_content: &[u8]) -> errors::Result<(Vec<u8>, Vec<u8>)> {
30+
let public_key = RsaPublicKey::from_pkcs1_der(key_content)
31+
.map_err(|e| ErrorKind::InvalidRsaKey(e.to_string()))?;
32+
Ok((public_key.n().to_bytes_be(), public_key.e().to_bytes_be()))
33+
}
34+
35+
fn ec_components_from_private_key(
2636
key_content: &[u8],
2737
alg: Algorithm,
2838
) -> errors::Result<(EllipticCurve, Vec<u8>, Vec<u8>)> {
@@ -108,9 +118,11 @@ fn new_verifier(
108118
pub static DEFAULT_PROVIDER: CryptoProvider = CryptoProvider {
109119
signer_factory: new_signer,
110120
verifier_factory: new_verifier,
111-
jwk_utils: JwkUtils {
112-
extract_rsa_public_key_components,
113-
extract_ec_public_key_coordinates,
121+
key_utils: KeyUtils {
122+
rsa_pub_components_from_private_key: rsa_components_from_private_key,
123+
rsa_pub_components_from_public_key: rsa_components_from_public_key,
124+
ec_pub_components_from_private_key: ec_components_from_private_key,
125+
ec_pub_components_from_public_key: crate::crypto::ec_components_from_public_key,
114126
compute_digest,
115127
},
116128
};

0 commit comments

Comments
 (0)