const ED25519_MULTICODEC: [u8; 2] = [0xED, 0x01];
#[derive(Debug, Clone, thiserror::Error, PartialEq, Eq)]
#[non_exhaustive]
pub enum DidKeyError {
#[error("DID must start with 'did:key:z', got: {0}")]
InvalidPrefix(String),
#[error("Base58 decoding failed: {0}")]
Base58DecodeFailed(String),
#[error("Unsupported or malformed multicodec: expected Ed25519 [0xED, 0x01]")]
UnsupportedMulticodec,
#[error("Invalid Ed25519 key length: expected 32 bytes, got {0}")]
InvalidKeyLength(usize),
}
impl crate::AuthsErrorInfo for DidKeyError {
fn error_code(&self) -> &'static str {
match self {
Self::InvalidPrefix(_) => "AUTHS-E1101",
Self::Base58DecodeFailed(_) => "AUTHS-E1102",
Self::UnsupportedMulticodec => "AUTHS-E1103",
Self::InvalidKeyLength(_) => "AUTHS-E1104",
}
}
fn suggestion(&self) -> Option<&'static str> {
match self {
Self::InvalidPrefix(_) => Some("DID must start with 'did:key:z'"),
Self::UnsupportedMulticodec => Some("Only Ed25519 keys are supported"),
_ => None,
}
}
}
pub fn did_key_to_ed25519(did: &str) -> Result<[u8; 32], DidKeyError> {
let encoded = strip_did_key_prefix(did)?;
let decoded = decode_base58(encoded)?;
validate_multicodec_and_extract(&decoded)
}
pub fn ed25519_pubkey_to_did_key(public_key: &[u8; 32]) -> String {
let mut prefixed = vec![0xED, 0x01];
prefixed.extend_from_slice(public_key);
let encoded = bs58::encode(prefixed).into_string();
format!("did:key:z{encoded}")
}
pub fn ed25519_pubkey_to_did_keri(pk: &[u8]) -> String {
format!("did:keri:{}", bs58::encode(pk).into_string())
}
fn strip_did_key_prefix(did: &str) -> Result<&str, DidKeyError> {
did.strip_prefix("did:key:z")
.ok_or_else(|| DidKeyError::InvalidPrefix(did.to_string()))
}
fn decode_base58(encoded: &str) -> Result<Vec<u8>, DidKeyError> {
bs58::decode(encoded)
.into_vec()
.map_err(|e| DidKeyError::Base58DecodeFailed(e.to_string()))
}
fn validate_multicodec_and_extract(decoded: &[u8]) -> Result<[u8; 32], DidKeyError> {
if decoded.len() != 34
|| decoded[0] != ED25519_MULTICODEC[0]
|| decoded[1] != ED25519_MULTICODEC[1]
{
if decoded.len() != 34 {
return Err(DidKeyError::InvalidKeyLength(
decoded.len().saturating_sub(2),
));
}
return Err(DidKeyError::UnsupportedMulticodec);
}
let mut key = [0u8; 32];
key.copy_from_slice(&decoded[2..]);
Ok(key)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn roundtrip_encode_decode() {
let original = [42u8; 32];
let did = ed25519_pubkey_to_did_key(&original);
assert!(did.starts_with("did:key:z"));
let decoded = did_key_to_ed25519(&did).unwrap();
assert_eq!(decoded, original);
}
#[test]
fn rejects_invalid_prefix() {
let err = did_key_to_ed25519("did:web:example.com").unwrap_err();
assert!(matches!(err, DidKeyError::InvalidPrefix(_)));
}
#[test]
fn rejects_invalid_base58() {
let err = did_key_to_ed25519("did:key:z0OOO").unwrap_err();
assert!(matches!(err, DidKeyError::Base58DecodeFailed(_)));
}
}