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 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
use std::collections::HashMap; use zeroize::Zeroize; use crate::identifiers::cred_def::CredentialDefinitionId; use crate::identifiers::rev_reg::RevocationRegistryId; use crate::identifiers::schema::SchemaId; use crate::{Validatable, ValidationError}; #[derive(Debug)] #[cfg_attr(feature = "serde", derive(Deserialize, Serialize))] pub struct Credential { pub schema_id: SchemaId, pub cred_def_id: CredentialDefinitionId, pub rev_reg_id: Option<RevocationRegistryId>, pub values: CredentialValues, pub signature: ursa_cl!(CredentialSignature), pub signature_correctness_proof: ursa_cl!(SignatureCorrectnessProof), pub rev_reg: Option<ursa_cl!(RevocationRegistry)>, pub witness: Option<ursa_cl!(Witness)>, } #[cfg(any(feature = "cl", feature = "cl_native"))] impl Credential { pub fn try_clone(&self) -> Result<Self, crate::ConversionError> { Ok(Self { schema_id: self.schema_id.clone(), cred_def_id: self.cred_def_id.clone(), rev_reg_id: self.rev_reg_id.clone(), values: self.values.clone(), signature: self.signature.try_clone()?, signature_correctness_proof: self.signature_correctness_proof.try_clone()?, rev_reg: self.rev_reg.clone(), witness: self.witness.clone(), }) } } impl Credential { pub const QUALIFIABLE_TAGS: [&'static str; 5] = [ "issuer_did", "cred_def_id", "schema_id", "schema_issuer_did", "rev_reg_id", ]; } impl Validatable for Credential { fn validate(&self) -> Result<(), ValidationError> { self.schema_id.validate()?; self.cred_def_id.validate()?; self.values.validate()?; if self.rev_reg_id.is_some() && (self.witness.is_none() || self.rev_reg.is_none()) { return Err("Credential validation failed: `witness` and `rev_reg` must be passed for revocable Credential".into()); } if self.values.0.is_empty() { return Err("Credential validation failed: `values` is empty".into()); } Ok(()) } } #[derive(Clone, Debug, PartialEq, Eq)] #[cfg_attr(feature = "serde", derive(Deserialize, Serialize))] pub struct CredentialInfo { pub referent: String, pub attrs: ShortCredentialValues, pub schema_id: SchemaId, pub cred_def_id: CredentialDefinitionId, pub rev_reg_id: Option<RevocationRegistryId>, pub cred_rev_id: Option<String>, } pub type ShortCredentialValues = HashMap<String, String>; #[derive(Clone, Debug, PartialEq, Eq)] #[cfg_attr(feature = "serde", derive(Deserialize, Serialize))] pub struct CredentialValues(pub HashMap<String, AttributeValues>); impl Drop for CredentialValues { fn drop(&mut self) { self.zeroize(); } } impl Validatable for CredentialValues { fn validate(&self) -> Result<(), ValidationError> { if self.0.is_empty() { return Err("CredentialValues validation failed: empty list has been passed".into()); } Ok(()) } } impl Zeroize for CredentialValues { fn zeroize(&mut self) { for attr in self.0.values_mut() { attr.zeroize(); } } } #[derive(Clone, Debug, PartialEq, Eq, Zeroize)] #[cfg_attr(feature = "serde", derive(Deserialize, Serialize))] pub struct AttributeValues { pub raw: String, pub encoded: String, }