iso14229-1 0.1.0

A ISO 14229-1 protocol.
Documentation
//! response of Service 2C

use crate::{
    error::Error,
    response::{Code, Response, SubFunction},
    DefinitionType, DynamicallyDID, Configuration, ResponseData, Service,
};
use std::{collections::HashSet, sync::LazyLock};

pub static DYNAMICAL_DID_NEGATIVES: LazyLock<HashSet<Code>> = LazyLock::new(|| {
    HashSet::from([
        Code::SubFunctionNotSupported,
        Code::IncorrectMessageLengthOrInvalidFormat,
        Code::ConditionsNotCorrect,
        Code::RequestOutOfRange,
        Code::SecurityAccessDenied,
    ])
});

#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub struct DynamicallyDefineDID(pub Option<DynamicallyDID>);

impl From<DynamicallyDefineDID> for Vec<u8> {
    fn from(v: DynamicallyDefineDID) -> Self {
        match v.0 {
            Some(v) => v.into(),
            None => vec![],
        }
    }
}

impl ResponseData for DynamicallyDefineDID {
    fn new_response<T: AsRef<[u8]>>(
        data: T,
        sub_func: Option<u8>,
        _: &Configuration,
    ) -> Result<Response, Error> {
        let data = data.as_ref();
        match sub_func {
            Some(sub_func) => {
                let _ = DefinitionType::try_from(sub_func)?;

                let data_len = data.len();
                match data_len {
                    0 | 2 => {}
                    _ => {
                        return Err(Error::InvalidDataLength {
                            expect: 0,
                            actual: data_len,
                        })
                    }
                }

                Ok(Response {
                    service: Service::DynamicalDefineDID,
                    negative: false,
                    sub_func: Some(SubFunction::new(sub_func)),
                    data: data.to_vec(),
                })
            }
            None => Err(Error::SubFunctionError(Service::DynamicalDefineDID)),
        }
    }
}

impl TryFrom<(&Response, &Configuration)> for DynamicallyDefineDID {
    type Error = Error;
    fn try_from((resp, _): (&Response, &Configuration)) -> Result<Self, Self::Error> {
        let service = resp.service;
        if service != Service::DynamicalDefineDID || resp.sub_func.is_none() {
            return Err(Error::ServiceError(service));
        }

        let data = &resp.data;
        let data_len = data.len();
        let offset = 0;

        let dynamic = match data_len {
            0 => Ok(None),
            2 => Ok(Some(DynamicallyDID::try_from(u16::from_be_bytes([
                data[offset],
                data[offset + 1],
            ]))?)),
            v => Err(Error::InvalidDataLength {
                expect: 2,
                actual: v,
            }),
        }?;

        Ok(Self(dynamic))
    }
}