core_crypto/mls_provider/
mod.rs1#![expect(unreachable_pub)]
3use std::sync::Arc;
4
5use async_lock::RwLock;
6pub use core_crypto_keystore::{Database, DatabaseKey};
7
8mod crypto_provider;
9mod error;
10
11pub(crate) use crypto_provider::CRYPTO;
12pub use crypto_provider::RustCrypto;
13pub use error::{Error, MlsProviderResult};
14use openmls_traits::{
15 authentication_service::{CredentialAuthenticationStatus, CredentialRef},
16 crypto::OpenMlsCrypto,
17 types::{
18 AeadType, Ciphersuite, CryptoError, ExporterSecret, HashType, HpkeCiphertext, HpkeConfig, HpkeKeyPair,
19 KemOutput, SignatureScheme,
20 },
21};
22#[allow(unused)]
24pub use wire_e2e_identity::pki::{CertProfile, CertificateGenerationArgs, PkiKeypair};
25use wire_e2e_identity::pki_env::PkiEnvironment;
26
27pub type RawEntropySeed = <rand_chacha::ChaCha20Rng as rand::SeedableRng>::Seed;
29
30#[derive(Debug, Clone, Default, PartialEq, Eq, zeroize::ZeroizeOnDrop)]
31#[repr(transparent)]
32pub struct EntropySeed(RawEntropySeed);
34
35impl EntropySeed {
36 pub const EXPECTED_LEN: usize = std::mem::size_of::<EntropySeed>() / std::mem::size_of::<u8>();
38
39 pub fn try_from_slice(data: &[u8]) -> MlsProviderResult<Self> {
41 if data.len() < Self::EXPECTED_LEN {
42 return Err(Error::EntropySeedLength {
43 actual: data.len(),
44 expected: Self::EXPECTED_LEN,
45 });
46 }
47
48 let mut inner = RawEntropySeed::default();
49 inner.copy_from_slice(&data[..Self::EXPECTED_LEN]);
50
51 Ok(Self(inner))
52 }
53
54 pub fn from_raw(raw: RawEntropySeed) -> Self {
56 Self(raw)
57 }
58}
59
60impl std::ops::Deref for EntropySeed {
61 type Target = [u8];
62 fn deref(&self) -> &Self::Target {
63 &self.0
64 }
65}
66
67impl std::ops::DerefMut for EntropySeed {
68 fn deref_mut(&mut self) -> &mut Self::Target {
69 &mut self.0
70 }
71}
72
73#[derive(Debug)]
74pub struct AuthenticationService {
75 pki_env: RwLock<Option<Arc<PkiEnvironment>>>,
84}
85
86impl AuthenticationService {
87 pub async fn pki_env(&self) -> Option<Arc<PkiEnvironment>> {
88 self.pki_env.read().await.clone()
89 }
90}
91
92#[cfg_attr(target_os = "unknown", async_trait::async_trait(?Send))]
93#[cfg_attr(not(target_os = "unknown"), async_trait::async_trait)]
94impl openmls_traits::authentication_service::AuthenticationServiceDelegate for AuthenticationService {
95 async fn validate_credential<'a>(&'a self, credential: CredentialRef<'a>) -> CredentialAuthenticationStatus {
96 match credential {
97 CredentialRef::Basic { .. } => CredentialAuthenticationStatus::Valid,
99
100 CredentialRef::X509 { .. } => match self.pki_env.read().await.as_ref() {
101 None => {
102 log::warn!("unable to validate X509 credentials: PKI environment is unset");
103 CredentialAuthenticationStatus::Unknown
104 }
105 Some(pki_env) => pki_env.validate_credential(credential).await,
106 },
107 }
108 }
109}
110
111#[derive(Debug, Clone)]
113pub struct CryptoProvider {
114 crypto: Arc<RustCrypto>,
115 key_store: Database,
116 auth_service: Arc<AuthenticationService>,
117}
118
119impl CryptoProvider {
120 pub fn new(key_store: Database) -> Self {
126 Self::new_with_pki_env(key_store, None)
127 }
128
129 pub fn new_with_pki_env(key_store: Database, pki_env: Option<Arc<PkiEnvironment>>) -> Self {
131 let pki_env = RwLock::new(pki_env);
132 let auth_service = Arc::new(AuthenticationService { pki_env });
133 Self {
134 key_store,
135 crypto: Arc::clone(&CRYPTO),
136 auth_service,
137 }
138 }
139
140 pub async fn new_transaction(&self) -> MlsProviderResult<()> {
143 self.key_store.new_transaction().await.map_err(Into::into)
144 }
145
146 pub async fn set_pki_environment(&mut self, pki_env: Option<Arc<PkiEnvironment>>) {
148 *self.auth_service.pki_env.write().await = pki_env;
149 }
150
151 pub async fn is_pki_env_setup(&self) -> bool {
153 self.auth_service.pki_env.read().await.is_some()
154 }
155
156 pub fn reseed(&self, entropy_seed: Option<EntropySeed>) -> MlsProviderResult<()> {
160 self.crypto.reseed(entropy_seed)
161 }
162
163 pub async fn close(&self) -> MlsProviderResult<()> {
167 self.key_store.close().await?;
168 Ok(())
169 }
170}
171
172impl openmls_traits::OpenMlsCryptoProvider for CryptoProvider {
173 type CryptoProvider = RustCrypto;
174 type RandProvider = RustCrypto;
175 type KeyStoreProvider = Database;
176 type AuthenticationServiceProvider = AuthenticationService;
177
178 fn crypto(&self) -> &Self::CryptoProvider {
179 &self.crypto
180 }
181
182 fn rand(&self) -> &Self::RandProvider {
183 &self.crypto
184 }
185
186 fn key_store(&self) -> &Self::KeyStoreProvider {
187 &self.key_store
188 }
189
190 fn authentication_service(&self) -> &Self::AuthenticationServiceProvider {
191 &self.auth_service
192 }
193}
194
195impl OpenMlsCrypto for &CryptoProvider {
197 fn supports(&self, ciphersuite: Ciphersuite) -> Result<(), CryptoError> {
198 self.crypto.supports(ciphersuite)
199 }
200
201 fn supported_ciphersuites(&self) -> Vec<Ciphersuite> {
202 self.crypto.supported_ciphersuites()
203 }
204
205 fn hkdf_extract(
206 &self,
207 hash_type: HashType,
208 salt: &[u8],
209 ikm: &[u8],
210 ) -> Result<tls_codec::SecretVLBytes, CryptoError> {
211 self.crypto.hkdf_extract(hash_type, salt, ikm)
212 }
213
214 fn hkdf_expand(
215 &self,
216 hash_type: HashType,
217 prk: &[u8],
218 info: &[u8],
219 okm_len: usize,
220 ) -> Result<tls_codec::SecretVLBytes, CryptoError> {
221 self.crypto.hkdf_expand(hash_type, prk, info, okm_len)
222 }
223
224 fn hash(&self, hash_type: HashType, data: &[u8]) -> Result<Vec<u8>, CryptoError> {
225 self.crypto.hash(hash_type, data)
226 }
227
228 fn aead_encrypt(
229 &self,
230 alg: AeadType,
231 key: &[u8],
232 data: &[u8],
233 nonce: &[u8],
234 aad: &[u8],
235 ) -> Result<Vec<u8>, CryptoError> {
236 self.crypto.aead_encrypt(alg, key, data, nonce, aad)
237 }
238
239 fn aead_decrypt(
240 &self,
241 alg: AeadType,
242 key: &[u8],
243 ct_tag: &[u8],
244 nonce: &[u8],
245 aad: &[u8],
246 ) -> Result<Vec<u8>, CryptoError> {
247 self.crypto.aead_decrypt(alg, key, ct_tag, nonce, aad)
248 }
249
250 fn signature_key_gen(&self, alg: SignatureScheme) -> Result<(Vec<u8>, Vec<u8>), CryptoError> {
251 self.crypto.signature_key_gen(alg)
252 }
253
254 fn signature_public_key_len(&self, alg: SignatureScheme) -> usize {
255 self.crypto.signature_public_key_len(alg)
256 }
257
258 fn validate_signature_key(&self, alg: SignatureScheme, key: &[u8]) -> Result<(), CryptoError> {
259 self.crypto.validate_signature_key(alg, key)
260 }
261
262 fn verify_signature(
263 &self,
264 alg: SignatureScheme,
265 data: &[u8],
266 pk: &[u8],
267 signature: &[u8],
268 ) -> Result<(), CryptoError> {
269 self.crypto.verify_signature(alg, data, pk, signature)
270 }
271
272 fn sign(&self, alg: SignatureScheme, data: &[u8], key: &[u8]) -> Result<Vec<u8>, CryptoError> {
273 self.crypto.sign(alg, data, key)
274 }
275
276 fn hpke_seal(
277 &self,
278 config: HpkeConfig,
279 pk_r: &[u8],
280 info: &[u8],
281 aad: &[u8],
282 ptxt: &[u8],
283 ) -> Result<HpkeCiphertext, CryptoError> {
284 self.crypto.hpke_seal(config, pk_r, info, aad, ptxt)
285 }
286
287 fn hpke_open(
288 &self,
289 config: HpkeConfig,
290 input: &HpkeCiphertext,
291 sk_r: &[u8],
292 info: &[u8],
293 aad: &[u8],
294 ) -> Result<Vec<u8>, CryptoError> {
295 self.crypto.hpke_open(config, input, sk_r, info, aad)
296 }
297
298 fn hpke_setup_sender_and_export(
299 &self,
300 config: HpkeConfig,
301 pk_r: &[u8],
302 info: &[u8],
303 exporter_context: &[u8],
304 exporter_length: usize,
305 ) -> Result<(KemOutput, ExporterSecret), CryptoError> {
306 self.crypto
307 .hpke_setup_sender_and_export(config, pk_r, info, exporter_context, exporter_length)
308 }
309
310 fn hpke_setup_receiver_and_export(
311 &self,
312 config: HpkeConfig,
313 enc: &[u8],
314 sk_r: &[u8],
315 info: &[u8],
316 exporter_context: &[u8],
317 exporter_length: usize,
318 ) -> Result<ExporterSecret, CryptoError> {
319 self.crypto
320 .hpke_setup_receiver_and_export(config, enc, sk_r, info, exporter_context, exporter_length)
321 }
322
323 fn derive_hpke_keypair(&self, config: HpkeConfig, ikm: &[u8]) -> Result<HpkeKeyPair, CryptoError> {
324 self.crypto.derive_hpke_keypair(config, ikm)
325 }
326}