keri-core 0.17.13

Core library for the Key Event Receipt Infrastructure
Documentation
use cesrox::{group::Group, primitives::IndexedSignature as CesrIndexedSignature};
use said::SelfAddressingIdentifier;
use serde::{Deserialize, Serialize};

use crate::{
    database::EventDatabase,
    error::Error,
    event::sections::seal::EventSeal,
    prefix::{BasicPrefix, IdentifierPrefix, IndexedSignature, SelfSigningPrefix},
    processor::event_storage::EventStorage,
};

use super::cesr_adapter::ParseError;

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq)]
pub enum Signature {
    /// Created by transferable identifier
    Transferable(SignerData, Vec<IndexedSignature>),
    /// Created by nontransferable identifier
    NonTransferable(Nontransferable),
}

#[derive(
    Serialize,
    Deserialize,
    Debug,
    Clone,
    PartialEq,
    rkyv::Archive,
    rkyv::Serialize,
    rkyv::Deserialize,
)]
#[rkyv(compare(PartialEq), derive(Debug))]
pub enum Nontransferable {
    Indexed(Vec<IndexedSignature>),
    Couplet(Vec<(BasicPrefix, SelfSigningPrefix)>),
}

#[derive(
    Serialize,
    Deserialize,
    Debug,
    Clone,
    PartialEq,
    rkyv::Serialize,
    rkyv::Deserialize,
    rkyv::Archive,
)]
#[rkyv(derive(Debug))]
pub enum Transferable {
    Seal(EventSeal, Vec<IndexedSignature>),
}

#[derive(Serialize, Deserialize, Debug, Clone, PartialEq)]
pub enum SignerData {
    EventSeal(EventSeal),
    LastEstablishment(IdentifierPrefix),
    JustSignatures,
}

impl SignerData {
    pub fn get_signer(&self) -> Option<IdentifierPrefix> {
        match self {
            SignerData::EventSeal(seal) => Some(seal.prefix.clone()),
            SignerData::LastEstablishment(id) => Some(id.clone()),
            SignerData::JustSignatures => None,
        }
    }
}

impl Signature {
    pub fn get_signer(&self) -> Option<IdentifierPrefix> {
        match self {
            Signature::Transferable(signer_data, _) => signer_data.get_signer(),
            Signature::NonTransferable(Nontransferable::Couplet(couplets)) => {
                // TODO
                Some(IdentifierPrefix::Basic(couplets[0].0.clone()))
            }
            Signature::NonTransferable(Nontransferable::Indexed(_)) => None,
        }
    }

    pub fn verify<D: EventDatabase>(
        &self,
        data: &[u8],
        storage: &EventStorage<D>,
    ) -> Result<bool, Error> {
        match self {
            Signature::Transferable(_sigd, sigs) => {
                let kc = storage
                    .get_state(&self.get_signer().ok_or(Error::MissingSigner)?)
                    .ok_or_else(|| Error::UnknownSigner(self.get_signer().unwrap()))?
                    .current;
                Ok(kc.verify(data, sigs)?)
            }
            Signature::NonTransferable(Nontransferable::Couplet(couplets)) => Ok(couplets
                .iter()
                .all(|(id, sig)| id.verify(data, sig).unwrap())),
            Signature::NonTransferable(Nontransferable::Indexed(_sigs)) => {
                Err(Error::MissingSigner)
            }
        }
    }
}

pub fn signatures_into_groups(sigs: &[Signature]) -> Vec<Group> {
    // Group same type of signature in one attachment
    let (trans_seal, trans_last, nontrans, indexed, witness_indexed) = sigs.iter().cloned().fold(
        (vec![], vec![], vec![], vec![], vec![]),
        |(mut trans_seal, mut trans_last, mut nontrans, mut indexed, mut witness_indexed), sig| {
            match sig {
                Signature::Transferable(SignerData::EventSeal(seal), sig) => {
                    let event_digest = seal.event_digest();
                    trans_seal.push((
                        seal.prefix.into(),
                        seal.sn,
                        event_digest.into(),
                        sig.into_iter().map(|sig| sig.into()).collect(),
                    ))
                }
                Signature::Transferable(SignerData::LastEstablishment(id), sig) => {
                    trans_last.push((id.into(), sig.into_iter().map(|sig| sig.into()).collect()))
                }
                Signature::Transferable(SignerData::JustSignatures, sig) => {
                    indexed.append(&mut sig.into_iter().map(|sig| sig.into()).collect())
                }
                Signature::NonTransferable(Nontransferable::Couplet(couplets)) => nontrans.append(
                    &mut couplets
                        .into_iter()
                        .map(|(bp, sp)| (bp.into(), sp.into()))
                        .collect(),
                ),
                Signature::NonTransferable(Nontransferable::Indexed(sigs)) => {
                    witness_indexed.append(&mut sigs.into_iter().map(|sig| sig.into()).collect())
                }
            };
            (trans_seal, trans_last, nontrans, indexed, witness_indexed)
        },
    );

    let mut attachments = vec![];
    if !trans_seal.is_empty() {
        attachments.push(Group::TransIndexedSigGroups(trans_seal));
    }
    if !trans_last.is_empty() {
        attachments.push(Group::LastEstSignaturesGroups(trans_last));
    }
    if !nontrans.is_empty() {
        attachments.push(Group::NontransReceiptCouples(nontrans));
    };
    if !indexed.is_empty() {
        attachments.push(Group::IndexedControllerSignatures(indexed));
    };
    if !witness_indexed.is_empty() {
        attachments.push(Group::IndexedWitnessSignatures(witness_indexed));
    };
    attachments
}

pub fn get_signatures(group: Group) -> Result<Vec<Signature>, ParseError> {
    match group {
        Group::IndexedControllerSignatures(sigs) => {
            let signatures = sigs.into_iter().map(|sig| sig.into()).collect();
            Ok(vec![Signature::Transferable(
                SignerData::JustSignatures,
                signatures,
            )])
        }
        Group::NontransReceiptCouples(sigs) => {
            let signatures = sigs
                .into_iter()
                .map(|(bp, sp)| (bp.into(), sp.into()))
                .collect();
            Ok(vec![Signature::NonTransferable(Nontransferable::Couplet(
                signatures,
            ))])
        }
        Group::LastEstSignaturesGroups(sigs) => Ok(sigs
            .into_iter()
            .map(|(id, sigs)| {
                let signatures = sigs.into_iter().map(|sig| sig.into()).collect();
                Signature::Transferable(SignerData::LastEstablishment(id.into()), signatures)
            })
            .collect()),
        Group::TransIndexedSigGroups(sigs) => Ok(sigs
            .into_iter()
            .map(|(id, sn, digest, sigs)| {
                let signatures = sigs.into_iter().map(|sig| sig.into()).collect();
                Signature::Transferable(
                    SignerData::EventSeal(EventSeal::new(
                        id.into(),
                        sn,
                        SelfAddressingIdentifier::from(digest),
                    )),
                    signatures,
                )
            })
            .collect()),
        Group::IndexedWitnessSignatures(sigs) => {
            let signatures = sigs.into_iter().map(|sig| sig.into()).collect();
            Ok(vec![Signature::NonTransferable(Nontransferable::Indexed(
                signatures,
            ))])
        }
        _ => Err(ParseError::AttachmentError(
            "Improper attachment type".into(),
        )),
    }
}

impl Into<Group> for Nontransferable {
    fn into(self) -> Group {
        match self {
            Nontransferable::Indexed(indexed) => {
                let signatures = indexed.into_iter().map(|sig| sig.into()).collect();
                Group::IndexedWitnessSignatures(signatures)
            }
            Nontransferable::Couplet(couples) => {
                let couples = couples
                    .into_iter()
                    .map(|(bp, sp)| (bp.into(), sp.into()))
                    .collect();
                Group::NontransReceiptCouples(couples)
            }
        }
    }
}

impl Into<Group> for crate::event_message::signature::Signature {
    fn into(self) -> Group {
        match self {
            crate::event_message::signature::Signature::Transferable(seal, signature) => {
                let signatures: Vec<CesrIndexedSignature> =
                    signature.into_iter().map(|sig| sig.into()).collect();
                match seal {
                    crate::event_message::signature::SignerData::EventSeal(event_seal) => {
                        let event_digest = event_seal.event_digest();
                        Group::TransIndexedSigGroups(vec![(
                            event_seal.prefix.into(),
                            event_seal.sn,
                            event_digest.into(),
                            signatures,
                        )])
                    }
                    crate::event_message::signature::SignerData::LastEstablishment(id) => {
                        Group::LastEstSignaturesGroups(vec![(id.into(), signatures)])
                    }
                    crate::event_message::signature::SignerData::JustSignatures => {
                        Group::IndexedControllerSignatures(signatures)
                    }
                }
            }
            crate::event_message::signature::Signature::NonTransferable(nt) => nt.into(),
        }
    }
}