use crate::{
asn1::*, ByteSlice, Choice, Decodable, Decoder, Encodable, Encoder, Error, ErrorKind, Header,
Length, Result, Tag,
};
use core::convert::{TryFrom, TryInto};
#[cfg(feature = "oid")]
use crate::asn1::ObjectIdentifier;
#[derive(Copy, Clone, Debug, Eq, PartialEq, PartialOrd, Ord)]
pub struct Any<'a> {
tag: Tag,
length: Length,
value: ByteSlice<'a>,
}
impl<'a> Any<'a> {
pub fn new(tag: Tag, bytes: &'a [u8]) -> Result<Self> {
let value = ByteSlice::new(bytes).map_err(|_| ErrorKind::Length { tag })?;
let length = if has_leading_zero_byte(tag) {
(value.len() + 1u8)?
} else {
value.len()
};
Ok(Self { tag, length, value })
}
pub(crate) fn from_tag_and_value(tag: Tag, value: ByteSlice<'a>) -> Self {
Self {
tag,
length: value.len(),
value,
}
}
pub fn tag(self) -> Tag {
self.tag
}
pub fn len(self) -> Length {
self.length
}
pub fn is_empty(self) -> bool {
self.value.is_empty()
}
pub fn is_null(self) -> bool {
Null::try_from(self).is_ok()
}
pub fn as_bytes(self) -> &'a [u8] {
self.value.as_bytes()
}
pub fn bit_string(self) -> Result<BitString<'a>> {
self.try_into()
}
pub fn context_specific(self) -> Result<ContextSpecific<'a>> {
self.try_into()
}
pub fn generalized_time(self) -> Result<GeneralizedTime> {
self.try_into()
}
pub fn ia5_string(self) -> Result<Ia5String<'a>> {
self.try_into()
}
pub fn octet_string(self) -> Result<OctetString<'a>> {
self.try_into()
}
#[cfg(feature = "oid")]
#[cfg_attr(docsrs, doc(cfg(feature = "oid")))]
pub fn oid(self) -> Result<ObjectIdentifier> {
self.try_into()
}
pub fn optional<T>(self) -> Result<Option<T>>
where
T: Choice<'a> + TryFrom<Self, Error = Error>,
{
if T::can_decode(self.tag) {
T::try_from(self).map(Some)
} else {
Ok(None)
}
}
pub fn printable_string(self) -> Result<PrintableString<'a>> {
self.try_into()
}
pub fn sequence<F, T>(self, f: F) -> Result<T>
where
F: FnOnce(&mut Decoder<'a>) -> Result<T>,
{
Sequence::try_from(self)?.decode_nested(f)
}
pub fn utc_time(self) -> Result<UtcTime> {
self.try_into()
}
pub fn utf8_string(self) -> Result<Utf8String<'a>> {
self.try_into()
}
}
impl<'a> Choice<'a> for Any<'a> {
fn can_decode(_: Tag) -> bool {
true
}
}
impl<'a> Decodable<'a> for Any<'a> {
fn decode(decoder: &mut Decoder<'a>) -> Result<Any<'a>> {
let header = Header::decode(decoder)?;
let tag = header.tag;
let mut value = decoder
.bytes(header.length)
.map_err(|_| decoder.error(ErrorKind::Length { tag }))?;
if has_leading_zero_byte(tag) {
let (byte, rest) = value
.split_first()
.ok_or(ErrorKind::Truncated)
.map_err(|e| decoder.error(e))?;
if *byte != 0 {
return Err(decoder.error(ErrorKind::Noncanonical { tag }));
}
value = rest;
}
Self::new(tag, value).map_err(|e| decoder.error(e.kind()))
}
}
impl<'a> Encodable for Any<'a> {
fn encoded_len(&self) -> Result<Length> {
self.len().for_tlv()
}
fn encode(&self, encoder: &mut Encoder<'_>) -> Result<()> {
Header::new(self.tag, self.len())?.encode(encoder)?;
if has_leading_zero_byte(self.tag) {
encoder.byte(0)?;
}
encoder.bytes(self.as_bytes())
}
}
impl<'a> TryFrom<&'a [u8]> for Any<'a> {
type Error = Error;
fn try_from(bytes: &'a [u8]) -> Result<Any<'a>> {
Any::from_der(bytes)
}
}
impl<'a> TryFrom<Any<'a>> for BitString<'a> {
type Error = Error;
fn try_from(any: Any<'a>) -> Result<BitString<'a>> {
any.tag().assert_eq(Tag::BitString)?;
Ok(BitString {
inner: any.value,
encoded_len: any.length,
})
}
}
impl<'a> From<BitString<'a>> for Any<'a> {
fn from(bit_string: BitString<'a>) -> Any<'a> {
Any {
tag: Tag::BitString,
length: bit_string.encoded_len,
value: bit_string.inner,
}
}
}
fn has_leading_zero_byte(tag: Tag) -> bool {
tag == Tag::BitString
}