use acdp_crypto::{verify_content_hash, verify_ecdsa_p256, verify_ed25519};
use acdp_primitives::error::AcdpError;
use acdp_types::body::{Body, Signature};
use acdp_types::primitives::ContentHash;
use acdp_types::publish::PublishRequest;
#[cfg(feature = "client")]
use {acdp_did::web::WebResolver, acdp_types::primitives::AgentDid};
#[cfg(feature = "client")]
pub struct Verifier<'a> {
resolver: &'a WebResolver,
}
#[cfg(feature = "client")]
impl<'a> Verifier<'a> {
pub fn new(resolver: &'a WebResolver) -> Self {
Self { resolver }
}
pub async fn verify_body(&self, body: &Body) -> Result<(), AcdpError> {
acdp_validation::validate_body(body)?;
self.verify_body_signed(body).await
}
pub async fn verify_body_signed(&self, body: &Body) -> Result<(), AcdpError> {
self.verify_body_hash(body)?;
self.verify_body_signature(body).await
}
pub fn verify_body_hash(&self, body: &Body) -> Result<(), AcdpError> {
let body_val = serde_json::to_value(body)?;
verify_content_hash(&body_val, &body.content_hash)
}
pub async fn verify_body_signature(&self, body: &Body) -> Result<(), AcdpError> {
verify_signature_envelope(
&body.agent_id,
&body.signature,
&body.content_hash,
self.resolver,
)
.await
}
}
#[cfg(feature = "client")]
pub async fn verify_publish_request_signature(
req: &PublishRequest,
resolver: &WebResolver,
) -> Result<(), AcdpError> {
verify_signature_envelope(&req.agent_id, &req.signature, &req.content_hash, resolver).await
}
#[cfg(feature = "client")]
async fn verify_signature_envelope(
agent_id: &AgentDid,
signature: &Signature,
content_hash: &ContentHash,
resolver: &WebResolver,
) -> Result<(), AcdpError> {
let key_id = &signature.key_id;
let (did_part, fragment) = key_id.split_once('#').ok_or_else(|| {
AcdpError::KeyResolution(format!("signature.key_id '{key_id}' has no '#fragment'"))
})?;
if fragment.is_empty() {
return Err(AcdpError::KeyResolution(format!(
"signature.key_id '{key_id}' has an empty '#fragment'"
)));
}
if did_part != agent_id.as_str() {
return Err(AcdpError::KeyNotAuthorized(format!(
"key_id DID '{did_part}' ≠ agent_id '{agent_id}'"
)));
}
if did_part.starts_with("did:key:") {
return verify_did_key_envelope(signature, content_hash);
}
if !did_part.starts_with("did:web:") {
return Err(AcdpError::KeyNotAuthorized(format!(
"signatures require a did:web or did:key key_id; got '{did_part}'"
)));
}
let doc = resolver.resolve(did_part).await?;
let method = doc.find_by_fragment(fragment).ok_or_else(|| {
AcdpError::KeyResolution(format!(
"no verification method with fragment '#{fragment}'"
))
})?;
if !doc.is_assertion_method(&method.id) {
return Err(AcdpError::KeyNotAuthorized(format!(
"'{}' is not in assertionMethod",
method.id
)));
}
if let Some(declared) = method.declared_algorithm() {
if declared != signature.algorithm {
return Err(AcdpError::InvalidSignature(format!(
"signature.algorithm '{}' does not match verification method type \
(resolved key declares '{declared}')",
signature.algorithm
)));
}
}
match signature.algorithm.as_str() {
"ed25519" => {
let pub_bytes = method.ed25519_public_key_bytes()?;
verify_ed25519(&pub_bytes, &signature.value, content_hash.as_str())
}
"ecdsa-p256" => {
let pub_sec1 = method.ecdsa_p256_public_key_sec1()?;
verify_ecdsa_p256(&pub_sec1, &signature.value, content_hash.as_str())
}
other => Err(AcdpError::UnsupportedAlgorithm(format!(
"verifier does not support signature algorithm '{other}'"
))),
}
}
pub fn verify_did_key_envelope(
signature: &Signature,
content_hash: &ContentHash,
) -> Result<(), AcdpError> {
let material = acdp_did::key::resolve_did_key_url(&signature.key_id)?;
if material.algorithm() != signature.algorithm {
return Err(AcdpError::InvalidSignature(format!(
"signature.algorithm '{}' does not match the did:key multicodec \
(key implies '{}')",
signature.algorithm,
material.algorithm()
)));
}
match material {
acdp_did::key::DidKeyMaterial::Ed25519(pub_bytes) => {
verify_ed25519(&pub_bytes, &signature.value, content_hash.as_str())
}
acdp_did::key::DidKeyMaterial::EcdsaP256(sec1_compressed) => {
verify_ecdsa_p256(&sec1_compressed, &signature.value, content_hash.as_str())
}
}
}
pub fn verify_body_offline(body: &Body) -> Result<(), AcdpError> {
acdp_validation::validate_body(body)?;
if !body.agent_id.as_str().starts_with("did:key:") {
return Err(AcdpError::KeyResolution(format!(
"verify_body_offline supports did:key producers only; '{}' requires \
the resolver-backed Verifier (client feature)",
body.agent_id
)));
}
let body_val = serde_json::to_value(body)?;
verify_content_hash(&body_val, &body.content_hash)?;
let did_part = body
.signature
.key_id
.split_once('#')
.map(|(d, _)| d)
.unwrap_or(body.signature.key_id.as_str());
if did_part != body.agent_id.as_str() {
return Err(AcdpError::KeyNotAuthorized(format!(
"key_id DID '{did_part}' ≠ agent_id '{}'",
body.agent_id
)));
}
verify_did_key_envelope(&body.signature, &body.content_hash)
}
pub fn verify_publish_request_signature_offline(req: &PublishRequest) -> Result<(), AcdpError> {
let key_id = req.signature.key_id.as_str();
let did_part = key_id.split_once('#').map(|(d, _)| d).unwrap_or(key_id);
if did_part != req.agent_id.as_str() {
return Err(AcdpError::KeyNotAuthorized(format!(
"key_id DID '{did_part}' ≠ agent_id '{}'",
req.agent_id
)));
}
if !did_part.starts_with("did:key:") {
return Err(AcdpError::KeyResolution(format!(
"offline verification supports did:key only; got '{did_part}'"
)));
}
verify_did_key_envelope(&req.signature, &req.content_hash)
}
#[cfg(feature = "client")]
pub async fn verify_body_signature_historical(
body: &Body,
resolver: &WebResolver,
) -> Result<(), AcdpError> {
let key_id = &body.signature.key_id;
let (did_part, fragment) = key_id.split_once('#').ok_or_else(|| {
AcdpError::KeyResolution(format!("signature.key_id '{key_id}' has no '#fragment'"))
})?;
if did_part != body.agent_id.as_str() {
return Err(AcdpError::KeyNotAuthorized(format!(
"key_id DID '{did_part}' ≠ agent_id '{}'",
body.agent_id
)));
}
if !did_part.starts_with("did:web:") {
return Err(AcdpError::KeyResolution(format!(
"historical-key verification applies to did:web only; got '{did_part}'"
)));
}
let doc = resolver.resolve(did_part).await?;
let method = doc.find_by_fragment(fragment).ok_or_else(|| {
AcdpError::KeyResolution(format!(
"no verification method with fragment '#{fragment}' — the key was \
removed from the DID document, not just rotated out of assertionMethod"
))
})?;
if let Some(declared) = method.declared_algorithm() {
if declared != body.signature.algorithm {
return Err(AcdpError::InvalidSignature(format!(
"signature.algorithm '{}' does not match verification method type \
(resolved key declares '{declared}')",
body.signature.algorithm
)));
}
}
match body.signature.algorithm.as_str() {
"ed25519" => verify_ed25519(
&method.ed25519_public_key_bytes()?,
&body.signature.value,
body.content_hash.as_str(),
),
"ecdsa-p256" => verify_ecdsa_p256(
&method.ecdsa_p256_public_key_sec1()?,
&body.signature.value,
body.content_hash.as_str(),
),
other => Err(AcdpError::UnsupportedAlgorithm(format!(
"verifier does not support signature algorithm '{other}'"
))),
}
}