use crate::{FixedTag, Header, Reader, Result, SliceReader};
use core::marker::PhantomData;
#[cfg(feature = "pem")]
use crate::{pem::PemLabel, PemReader};
#[cfg(doc)]
use crate::{Length, Tag};
#[cfg(feature = "alloc")]
use alloc::boxed::Box;
pub trait Decode<'a>: Sized {
fn decode<R: Reader<'a>>(decoder: &mut R) -> Result<Self>;
fn from_der(bytes: &'a [u8]) -> Result<Self> {
let mut reader = SliceReader::new(bytes)?;
let result = Self::decode(&mut reader)?;
reader.finish(result)
}
}
impl<'a, T> Decode<'a> for T
where
T: DecodeValue<'a> + FixedTag,
{
fn decode<R: Reader<'a>>(reader: &mut R) -> Result<T> {
let header = Header::decode(reader)?;
header.tag.assert_eq(T::TAG)?;
T::decode_value(reader, header)
}
}
impl<'a, T> Decode<'a> for PhantomData<T>
where
T: ?Sized,
{
fn decode<R: Reader<'a>>(_reader: &mut R) -> Result<PhantomData<T>> {
Ok(PhantomData)
}
}
pub trait DecodeOwned: for<'a> Decode<'a> {}
impl<T> DecodeOwned for T where T: for<'a> Decode<'a> {}
#[cfg(feature = "pem")]
pub trait DecodePem: DecodeOwned + PemLabel {
fn from_pem(pem: impl AsRef<[u8]>) -> Result<Self>;
}
#[cfg(feature = "pem")]
impl<T: DecodeOwned + PemLabel> DecodePem for T {
fn from_pem(pem: impl AsRef<[u8]>) -> Result<Self> {
let mut reader = PemReader::new(pem.as_ref())?;
Self::validate_pem_label(reader.type_label())?;
T::decode(&mut reader)
}
}
pub trait DecodeValue<'a>: Sized {
fn decode_value<R: Reader<'a>>(reader: &mut R, header: Header) -> Result<Self>;
}
#[cfg(feature = "alloc")]
impl<'a, T> DecodeValue<'a> for Box<T>
where
T: DecodeValue<'a>,
{
fn decode_value<R: Reader<'a>>(reader: &mut R, header: Header) -> Result<Self> {
Ok(Box::new(T::decode_value(reader, header)?))
}
}