use crate::error::AcdpError;
use base64::{engine::general_purpose::STANDARD, Engine};
use ed25519_dalek::{Verifier as _, VerifyingKey};
#[cfg(feature = "client")]
use {
super::hash::verify_content_hash,
crate::did::web::WebResolver,
crate::types::{
body::{Body, Signature},
primitives::{AgentDid, ContentHash},
publish::PublishRequest,
},
};
#[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> {
crate::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.starts_with("did:web:") {
return Err(AcdpError::KeyNotAuthorized(format!(
"v0.1.0 signatures require did:web key_id; got '{did_part}'"
)));
}
if did_part != agent_id.as_str() {
return Err(AcdpError::KeyNotAuthorized(format!(
"key_id DID '{did_part}' ≠ agent_id '{agent_id}'"
)));
}
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_ed25519(
pub_key_bytes: &[u8; 32],
sig_b64: &str,
message: &str,
) -> Result<(), AcdpError> {
let key = VerifyingKey::from_bytes(pub_key_bytes)
.map_err(|e| AcdpError::InvalidSignature(e.to_string()))?;
let sig_bytes = STANDARD
.decode(sig_b64)
.map_err(|e| AcdpError::InvalidSignature(format!("base64: {e}")))?;
let sig = ed25519_dalek::Signature::from_slice(&sig_bytes)
.map_err(|e| AcdpError::InvalidSignature(format!("sig parse: {e}")))?;
key.verify(message.as_bytes(), &sig)
.map_err(|_| AcdpError::InvalidSignature("signature verification failed".into()))
}
pub fn verify_ecdsa_p256(
pub_key_sec1: &[u8],
sig_b64: &str,
message: &str,
) -> Result<(), AcdpError> {
use p256::ecdsa::{signature::Verifier as _, Signature, VerifyingKey as P256VerifyingKey};
let key = P256VerifyingKey::from_sec1_bytes(pub_key_sec1)
.map_err(|e| AcdpError::InvalidSignature(format!("ecdsa-p256 key parse: {e}")))?;
let sig_bytes = STANDARD
.decode(sig_b64)
.map_err(|e| AcdpError::InvalidSignature(format!("base64: {e}")))?;
if sig_bytes.len() != 64 {
return Err(AcdpError::InvalidSignature(format!(
"ecdsa-p256 signature MUST be 64 bytes (IEEE 1363 r‖s), got {}",
sig_bytes.len()
)));
}
let sig = Signature::from_slice(&sig_bytes)
.map_err(|e| AcdpError::InvalidSignature(format!("ecdsa-p256 sig parse: {e}")))?;
key.verify(message.as_bytes(), &sig)
.map_err(|_| AcdpError::InvalidSignature("ecdsa-p256 signature verification failed".into()))
}
#[cfg(test)]
mod tests {
use super::*;
use crate::crypto::sign::SigningKey;
use crate::types::primitives::ContentHash;
const TEST_SEED: [u8; 32] = [0u8; 32];
const TEST_PUB_HEX: &str = "3b6a27bcceb6a42d62a3a8d02a6f0d73653215771de243a63ac048a18b59da29";
#[test]
fn sign_and_verify_golden() {
let key = SigningKey::from_bytes(&TEST_SEED);
let hash = ContentHash(
"sha256:f170150ddbf59d99794e7797824591b374d459782084597b644ecc57a41031b5".into(),
);
let sig_b64 = key.sign_content_hash(&hash);
assert_eq!(
sig_b64,
"ErkbV+FUdn49TgF3zJ3RBe3AmyGxLVAQdMjlhabUfM96qendmWwdVodX/SV3O3aKLypbUu6gmb5Npt3O/w7nDQ=="
);
let pub_bytes: [u8; 32] = hex::decode(TEST_PUB_HEX).unwrap().try_into().unwrap();
verify_ed25519(&pub_bytes, &sig_b64, hash.as_str()).unwrap();
}
#[test]
fn wrong_message_fails() {
let key = SigningKey::from_bytes(&TEST_SEED);
let hash = ContentHash(
"sha256:f170150ddbf59d99794e7797824591b374d459782084597b644ecc57a41031b5".into(),
);
let sig_b64 = key.sign_content_hash(&hash);
let pub_bytes: [u8; 32] = hex::decode(TEST_PUB_HEX).unwrap().try_into().unwrap();
let result = verify_ed25519(&pub_bytes, &sig_b64, "sha256:wronghash");
assert!(result.is_err());
}
#[test]
fn declared_algorithm_mismatch_rejected() {
use crate::did::document::VerificationMethod;
let raw: [u8; 32] = hex::decode(TEST_PUB_HEX).unwrap().try_into().unwrap();
let mut prefixed = vec![0xed, 0x01];
prefixed.extend_from_slice(&raw);
let mb = format!("z{}", bs58::encode(&prefixed).into_string());
let vm = VerificationMethod {
id: "did:web:example.com#key-1".into(),
method_type: "Ed25519VerificationKey2020".into(),
controller: "did:web:example.com".into(),
public_key_jwk: None,
public_key_multibase: Some(mb),
};
assert_eq!(vm.declared_algorithm(), Some("ed25519"));
}
}