1use std::{
12 ffi::{c_uint, CStr, CString},
13 fmt::Debug,
14};
15
16use libcoap_sys::{
17 coap_const_char_ptr_t, coap_dtls_key_t, coap_dtls_key_t__bindgen_ty_1, coap_dtls_pki_t, coap_pki_define_t,
18 coap_pki_define_t_COAP_PKI_KEY_DEF_DER, coap_pki_define_t_COAP_PKI_KEY_DEF_DER_BUF,
19 coap_pki_define_t_COAP_PKI_KEY_DEF_ENGINE, coap_pki_define_t_COAP_PKI_KEY_DEF_PEM,
20 coap_pki_define_t_COAP_PKI_KEY_DEF_PEM_BUF, coap_pki_define_t_COAP_PKI_KEY_DEF_PKCS11, coap_pki_key_define_t,
21 coap_pki_key_t_COAP_PKI_KEY_DEFINE,
22};
23
24use crate::{
25 crypto::{
26 pki_rpk,
27 pki_rpk::{
28 key::{KeyComponentSealed, KeyTypeSealed},
29 Asn1PrivateKeyType, CertVerificationMode, CertVerifying, CnCallback, DerFileKeyComponent,
30 DerMemoryKeyComponent, EngineKeyComponent, KeyComponent, KeyDef, KeyDefSealed, NonCertVerifying,
31 PemFileKeyComponent, PemMemoryKeyComponent, Pkcs11KeyComponent, PkiRpkContext, PkiRpkContextBuilder,
32 ServerPkiRpkCryptoContext,
33 },
34 ClientCryptoContext,
35 },
36 session::CoapSession,
37};
38
39#[derive(Debug, Clone, Copy)]
41pub struct Pki {}
42
43impl KeyTypeSealed for Pki {
44 fn set_key_type_defaults(ctx: &mut coap_dtls_pki_t) {
45 ctx.is_rpk_not_cert = 0;
46 }
47}
48
49impl<'a> From<PkiRpkContext<'a, Pki>> for ServerPkiRpkCryptoContext<'a> {
53 fn from(value: PkiRpkContext<'a, Pki>) -> Self {
54 ServerPkiRpkCryptoContext::Pki(value)
55 }
56}
57
58impl<'a> From<PkiRpkContext<'a, Pki>> for ClientCryptoContext<'a> {
59 fn from(value: PkiRpkContext<'a, Pki>) -> Self {
60 ClientCryptoContext::Pki(value)
61 }
62}
63
64impl<'a> PkiRpkContextBuilder<'a, Pki, NonCertVerifying> {
65 pub fn verify_peer_cert(mut self) -> PkiRpkContextBuilder<'a, Pki, CertVerifying> {
81 self.ctx.raw_cfg.verify_peer_cert = 1;
82 PkiRpkContextBuilder::<Pki, CertVerifying> {
83 ctx: self.ctx,
84 verifying: Default::default(),
85 }
86 }
87}
88
89impl<'a> PkiRpkContextBuilder<'a, Pki, CertVerifying> {
90 pub fn new<K: KeyDef<KeyType = Pki> + 'a>(key: K) -> Self {
91 PkiRpkContextBuilder::<'a, Pki, NonCertVerifying>::new(key).verify_peer_cert()
92 }
93}
94
95impl<'a, V: CertVerificationMode> PkiRpkContextBuilder<'a, Pki, V> {
96 pub fn cn_validator(mut self, validator: impl PkiCnValidator + 'a) -> Self {
107 self.ctx.cn_callback = Some(CnCallback::Pki(Box::new(validator)));
108 self.ctx.raw_cfg.validate_cn_call_back = Some(pki_rpk::dtls_pki_cn_callback::<Pki>);
109 self
110 }
111}
112
113pub trait PkiCnValidator {
116 fn validate_cn(
130 &self,
131 cn: &CStr,
132 asn1_public_cert: &[u8],
133 session: &CoapSession,
134 depth: c_uint,
135 validated: bool,
136 ) -> bool;
137}
138
139impl<T: Fn(&CStr, &[u8], &CoapSession, c_uint, bool) -> bool> PkiCnValidator for T {
140 fn validate_cn(
141 &self,
142 cn: &CStr,
143 asn1_public_cert: &[u8],
144 session: &CoapSession,
145 depth: c_uint,
146 validated: bool,
147 ) -> bool {
148 self(cn, asn1_public_cert, session, depth, validated)
149 }
150}
151
152#[derive(Clone, Debug)]
175pub struct PkiKeyDef<CA: KeyComponent<Pki>, PK: KeyComponent<Pki>, SK: KeyComponent<Pki>> {
176 ca_cert: Option<CA>,
177 public_cert: PK,
178 private_key: SK,
179 user_pin: Option<CString>,
180 asn1_private_key_type: Asn1PrivateKeyType,
181}
182
183impl<CA: KeyComponent<Pki>, PK: KeyComponent<Pki>, SK: KeyComponent<Pki>> PkiKeyDef<CA, PK, SK> {
184 pub fn new(
199 ca_cert: Option<CA>,
200 public_cert: PK,
201 private_key: SK,
202 user_pin: Option<CString>,
203 asn1_private_key_type: Asn1PrivateKeyType,
204 ) -> Self {
205 Self {
206 ca_cert,
207 public_cert,
208 private_key,
209 user_pin,
210 asn1_private_key_type,
211 }
212 }
213}
214
215impl PkiKeyDef<PemFileKeyComponent, PemFileKeyComponent, PemFileKeyComponent> {
216 pub fn with_pem_files(
221 ca_cert: Option<impl Into<PemFileKeyComponent>>,
222 public_cert: impl Into<PemFileKeyComponent>,
223 private_key: impl Into<PemFileKeyComponent>,
224 ) -> Self {
225 Self::new(
226 ca_cert.map(|v| v.into()),
227 public_cert.into(),
228 private_key.into(),
229 None,
230 Asn1PrivateKeyType::None,
231 )
232 }
233}
234
235impl PkiKeyDef<PemMemoryKeyComponent, PemMemoryKeyComponent, PemMemoryKeyComponent> {
236 pub fn with_pem_memory(
241 ca_cert: Option<impl Into<PemMemoryKeyComponent>>,
242 public_cert: impl Into<PemMemoryKeyComponent>,
243 private_key: impl Into<PemMemoryKeyComponent>,
244 ) -> Self {
245 Self::new(
246 ca_cert.map(|v| v.into()),
247 public_cert.into(),
248 private_key.into(),
249 None,
250 Asn1PrivateKeyType::None,
251 )
252 }
253}
254
255impl PkiKeyDef<DerFileKeyComponent, DerFileKeyComponent, DerFileKeyComponent> {
256 pub fn with_asn1_files(
261 ca_cert: Option<impl Into<DerFileKeyComponent>>,
262 public_cert: impl Into<DerFileKeyComponent>,
263 private_key: impl Into<DerFileKeyComponent>,
264 private_key_type: Asn1PrivateKeyType,
265 ) -> Self {
266 Self::new(
267 ca_cert.map(|v| v.into()),
268 public_cert.into(),
269 private_key.into(),
270 None,
271 private_key_type,
272 )
273 }
274}
275
276impl PkiKeyDef<DerMemoryKeyComponent, DerMemoryKeyComponent, DerMemoryKeyComponent> {
277 pub fn with_asn1_memory(
282 ca_cert: Option<impl Into<DerMemoryKeyComponent>>,
283 public_cert: impl Into<DerMemoryKeyComponent>,
284 private_key: impl Into<DerMemoryKeyComponent>,
285 private_key_type: Asn1PrivateKeyType,
286 ) -> Self {
287 Self::new(
288 ca_cert.map(|v| v.into()),
289 public_cert.into(),
290 private_key.into(),
291 None,
292 private_key_type,
293 )
294 }
295}
296
297impl PkiKeyDef<Pkcs11KeyComponent, Pkcs11KeyComponent, Pkcs11KeyComponent> {
298 pub fn with_pkcs11(
303 ca_cert: Option<impl Into<Pkcs11KeyComponent>>,
304 public_cert: impl Into<Pkcs11KeyComponent>,
305 private_key: impl Into<Pkcs11KeyComponent>,
306 user_pin: Option<CString>,
307 ) -> Self {
308 Self::new(
309 ca_cert.map(|v| v.into()),
310 public_cert.into(),
311 private_key.into(),
312 user_pin,
313 Asn1PrivateKeyType::None,
314 )
315 }
316}
317
318impl<CA: KeyComponent<Pki>, PK: KeyComponent<Pki>, SK: KeyComponent<Pki>> KeyDefSealed for PkiKeyDef<CA, PK, SK> {
319 fn as_raw_dtls_key(&self) -> coap_dtls_key_t {
320 let (ca, ca_len) = self.ca_cert.as_ref().map(|v| v.as_raw_key_component()).unwrap_or((
321 coap_const_char_ptr_t {
322 u_byte: std::ptr::null(),
323 },
324 0,
325 ));
326 let (public_cert, public_cert_len) = self.public_cert.as_raw_key_component();
327 let (private_key, private_key_len) = self.private_key.as_raw_key_component();
328
329 coap_dtls_key_t {
330 key_type: coap_pki_key_t_COAP_PKI_KEY_DEFINE,
331 key: coap_dtls_key_t__bindgen_ty_1 {
332 define: coap_pki_key_define_t {
333 ca,
334 public_cert,
335 private_key,
336 ca_len,
337 public_cert_len,
338 private_key_len,
339 ca_def: <CA as KeyComponentSealed<Pki>>::DEFINE_TYPE,
340 public_cert_def: <PK as KeyComponentSealed<Pki>>::DEFINE_TYPE,
341 private_key_def: <SK as KeyComponentSealed<Pki>>::DEFINE_TYPE,
342 private_key_type: self.asn1_private_key_type.into(),
343 user_pin: self.user_pin.as_ref().map(|v| v.as_ptr()).unwrap_or(std::ptr::null()),
344 },
345 },
346 }
347 }
348}
349
350impl<CA: KeyComponent<Pki>, PK: KeyComponent<Pki>, SK: KeyComponent<Pki>> KeyDef for PkiKeyDef<CA, PK, SK> {
351 type KeyType = Pki;
352}
353
354impl KeyComponentSealed<Pki> for PemFileKeyComponent {
355 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_PEM;
356}
357
358impl KeyComponentSealed<Pki> for PemMemoryKeyComponent {
359 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_PEM_BUF;
360}
361
362impl KeyComponentSealed<Pki> for DerFileKeyComponent {
363 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_DER;
364}
365
366impl KeyComponentSealed<Pki> for DerMemoryKeyComponent {
367 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_DER_BUF;
368}
369
370impl KeyComponentSealed<Pki> for Pkcs11KeyComponent {
371 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_PKCS11;
372}
373
374impl KeyComponentSealed<Pki> for EngineKeyComponent {
375 const DEFINE_TYPE: coap_pki_define_t = coap_pki_define_t_COAP_PKI_KEY_DEF_ENGINE;
376}