1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82
//! `enveloped-data` content type [RFC 5652 § 6](https://datatracker.ietf.org/doc/html/rfc5652#section-6)
use crate::ContentType;
use der::{
asn1::{ContextSpecific, OctetString},
Decodable, Decoder, Encodable, Sequence, TagMode, TagNumber,
};
use spki::AlgorithmIdentifier;
type ContentEncryptionAlgorithmIdentifier<'a> = AlgorithmIdentifier<'a>;
const ENCRYPTED_CONTENT_TAG: TagNumber = TagNumber::new(0);
/// Encrypted content information [RFC 5652 § 6](https://datatracker.ietf.org/doc/html/rfc5652#section-6)
///
/// ```text
/// EncryptedContentInfo ::= SEQUENCE {
/// contentType ContentType,
/// contentEncryptionAlgorithm
/// ContentEncryptionAlgorithmIdentifier,
/// encryptedContent
/// [0] IMPLICIT EncryptedContent OPTIONAL }
///
/// ContentEncryptionAlgorithmIdentifier ::=
/// AlgorithmIdentifier
///
/// EncryptedContent ::= OCTET STRING
/// ```
///
/// The fields of type `EncryptedContentInfo` have the following meanings:
/// - [`content_type`](EncryptedContentInfo::content_type) indicates the type of content.
/// - [`content_encryption_algorithm`](EncryptedContentInfo::content_encryption_algorithm)
/// identifies the content-encryption algorithm (and any associated parameters) under
/// which the content is encrypted.
/// This algorithm is the same for all recipients.
/// - [`encrypted_content`](EncryptedContentInfo::encrypted_content) is the result of
/// encrypting the content. The field is optional, and if the field is not present,
/// its intended value must be supplied by other means.
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub struct EncryptedContentInfo<'a> {
/// indicates the type of content.
pub content_type: ContentType,
/// identifies the content-encryption algorithm (and any associated parameters) under
/// which the content is encrypted.
pub content_encryption_algorithm: ContentEncryptionAlgorithmIdentifier<'a>,
/// the encrypted contents;
/// when not present, its intended value must be supplied by other means.
pub encrypted_content: Option<&'a [u8]>,
}
impl<'a> Decodable<'a> for EncryptedContentInfo<'a> {
fn decode(decoder: &mut Decoder<'a>) -> der::Result<EncryptedContentInfo<'a>> {
decoder.sequence(|decoder| {
Ok(EncryptedContentInfo {
content_type: decoder.decode()?,
content_encryption_algorithm: decoder.decode()?,
encrypted_content: decoder
.context_specific::<OctetString<'_>>(ENCRYPTED_CONTENT_TAG, TagMode::Implicit)?
.map(|o| o.as_bytes()),
})
})
}
}
impl<'a> Sequence<'a> for EncryptedContentInfo<'a> {
fn fields<F, T>(&self, f: F) -> der::Result<T>
where
F: FnOnce(&[&dyn Encodable]) -> der::Result<T>,
{
let opt_octet = self.encrypted_content.map(OctetString::new).transpose()?;
f(&[
&self.content_type,
&self.content_encryption_algorithm,
&opt_octet.as_ref().map(|d| ContextSpecific {
tag_number: ENCRYPTED_CONTENT_TAG,
tag_mode: TagMode::Implicit,
value: *d,
}),
])
}
}