use std::collections::HashMap;
use affinidi_encoding::EncodingError;
use serde::{Deserialize, Serialize};
use serde_json::Value;
use thiserror::Error;
use url::Url;
use crate::{
service::Service,
verification_method::{VerificationMethod, VerificationRelationship},
};
pub mod builder;
pub mod did;
pub mod did_method;
pub mod document;
pub mod one_or_many;
pub mod service;
pub mod verification_method;
pub use builder::{DocumentBuilder, ServiceBuilder, VerificationMethodBuilder};
pub use did::{DID, DIDError};
pub use did_method::DIDMethod;
pub use did_method::key::{KeyError, KeyMaterial, KeyMaterialFormat, KeyMaterialType};
pub use did_method::peer::{
PeerCreateKey, PeerCreatedKey, PeerError, PeerKeyPurpose, PeerKeyType, PeerNumAlgo,
PeerPurpose, PeerService, PeerServiceEndpoint, PeerServiceEndpointLong,
PeerServiceEndpointShort,
};
pub use document::DocumentExt;
#[derive(Error, Debug)]
pub enum DocumentError {
#[error("URL Error")]
URL(#[from] url::ParseError),
#[error("VerificationMethod Error: {0}")]
VM(String),
#[error("Encoding Error: {0}")]
Encoding(#[from] EncodingError),
#[error("Key expansion error: {0}")]
KeyExpansionError(String),
}
#[derive(Debug, Serialize, Deserialize, Clone, PartialEq, Eq)]
#[serde(rename_all = "camelCase")]
pub struct Document {
pub id: Url,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub verification_method: Vec<VerificationMethod>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub authentication: Vec<VerificationRelationship>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub assertion_method: Vec<VerificationRelationship>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub key_agreement: Vec<VerificationRelationship>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub capability_invocation: Vec<VerificationRelationship>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub capability_delegation: Vec<VerificationRelationship>,
#[serde(skip_serializing_if = "Vec::is_empty", default)]
pub service: Vec<Service>,
#[serde(flatten)]
pub parameters_set: HashMap<String, Value>,
}
impl Default for Document {
fn default() -> Self {
Self {
id: Url::parse("did:example:123456789abcdefghi").unwrap(),
verification_method: Vec::new(),
authentication: Vec::new(),
assertion_method: Vec::new(),
key_agreement: Vec::new(),
capability_invocation: Vec::new(),
capability_delegation: Vec::new(),
service: Vec::new(),
parameters_set: HashMap::new(),
}
}
}
impl Document {
pub fn new(id: &str) -> Result<Self, DocumentError> {
Ok(Document {
id: Url::parse(id)?,
..Default::default()
})
}
}
#[cfg(test)]
mod tests {
use super::*;
use url::Url;
#[test]
fn valid_id() {
assert!(Url::parse("did:example:123456789abcdefghi").is_ok());
assert!(Url::parse("did:webvh:Qmd1FCL9Vj2vJ433UDfC9MBstK6W6QWSQvYyeNn8va2fai:identity.foundation:didwebvh-implementations:implementations:affinidi-didwebvh-rs").is_ok());
}
#[test]
fn document_new_valid() {
let doc = Document::new("did:example:123").unwrap();
assert_eq!(doc.id.as_str(), "did:example:123");
assert!(doc.verification_method.is_empty());
assert!(doc.service.is_empty());
}
#[test]
fn document_new_invalid() {
assert!(Document::new("not a url").is_err());
}
#[test]
fn document_default_has_example_id() {
let doc = Document::default();
assert_eq!(doc.id.as_str(), "did:example:123456789abcdefghi");
}
#[test]
fn document_serde_roundtrip_minimal() {
let doc = Document::new("did:example:456").unwrap();
let json = serde_json::to_string(&doc).unwrap();
let back: Document = serde_json::from_str(&json).unwrap();
assert_eq!(doc, back);
}
}