Skip to content

Commit

Permalink
Fix ed25519 key format in pkcs8
Browse files Browse the repository at this point in the history
* Add support for decoding ed25519 keys in PrivateKeyInfo format
* Fix ed25519 encoding and decoding in OneAsymmetricKey format

Closes: #271
  • Loading branch information
robertabcd committed Apr 26, 2024
1 parent 3041b0c commit 120c1a0
Show file tree
Hide file tree
Showing 2 changed files with 55 additions and 14 deletions.
37 changes: 23 additions & 14 deletions russh-keys/src/format/pkcs8.rs
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ fn write_key_v1(writer: &mut yasna::DERWriterSeq, secret: &ed25519_dalek::Signin
writer.next().write_bytes(&seed);
writer
.next()
.write_tagged(yasna::Tag::context(1), |writer| {
.write_tagged_implicit(yasna::Tag::context(1), |writer| {
writer.write_bitvec(&BitVec::from_bytes(public.as_bytes()))
})
}
Expand All @@ -150,24 +150,29 @@ fn read_key_v1(reader: &mut BERReaderSeq) -> Result<key::KeyPair, Error> {
.next()
.read_sequence(|reader| reader.next().read_oid())?;
if oid.components().as_slice() == ED25519 {
use ed25519_dalek::SigningKey;
let secret = {
let s = yasna::parse_der(&reader.next().read_bytes()?, |reader| reader.read_bytes())?;

s.get(..ed25519_dalek::SECRET_KEY_LENGTH)
.ok_or(Error::KeyIsCorrupt)
.and_then(|s| SigningKey::try_from(s).map_err(|_| Error::CouldNotReadKey))?
};
// Consume the public key
reader
.next()
.read_tagged(yasna::Tag::context(1), |reader| reader.read_bitvec())?;
Ok(key::KeyPair::Ed25519(secret))
let key = parse_ed25519_private_key(&reader.next().read_bytes()?)?;
let _attributes = reader.read_optional(|reader| {
reader.read_tagged_implicit(yasna::Tag::context(0), |reader| reader.read_der())
})?;
let _public_key = reader.read_optional(|reader| {
reader.read_tagged_implicit(yasna::Tag::context(1), |reader| reader.read_bitvec())
})?;
Ok(key)
} else {
Err(Error::CouldNotReadKey)
}
}

fn parse_ed25519_private_key(der: &[u8]) -> Result<key::KeyPair, Error> {
use ed25519_dalek::SigningKey;
let secret_bytes = yasna::parse_der(der, |reader| reader.read_bytes())?;
let secret_key = secret_bytes
.get(..ed25519_dalek::SECRET_KEY_LENGTH)
.ok_or(Error::KeyIsCorrupt)
.and_then(|s| SigningKey::try_from(s).map_err(|_| Error::CouldNotReadKey))?;
Ok(key::KeyPair::Ed25519(secret_key))
}

#[cfg(feature = "openssl")]
fn write_key_v0_rsa(writer: &mut yasna::DERWriterSeq, key: &Rsa<Private>) {
writer.next().write_u32(0);
Expand Down Expand Up @@ -242,6 +247,8 @@ fn write_key_v0_ec(writer: &mut yasna::DERWriterSeq, key: &crate::ec::PrivateKey
// Utility enum used for reading v0 key.
enum KeyType {
Unknown(ObjectIdentifier),
#[allow(clippy::upper_case_acronyms)]
ED25519,
#[cfg(feature = "openssl")]
#[allow(clippy::upper_case_acronyms)]
RSA,
Expand All @@ -252,6 +259,7 @@ fn read_key_v0(reader: &mut BERReaderSeq) -> Result<key::KeyPair, Error> {
let key_type = reader.next().read_sequence(|reader| {
let oid = reader.next().read_oid()?;
Ok(match oid.components().as_slice() {
ED25519 => KeyType::ED25519,
#[cfg(feature = "openssl")]
RSA => {
reader.next().read_null()?;
Expand All @@ -263,6 +271,7 @@ fn read_key_v0(reader: &mut BERReaderSeq) -> Result<key::KeyPair, Error> {
})?;
match key_type {
KeyType::Unknown(_) => Err(Error::CouldNotReadKey),
KeyType::ED25519 => parse_ed25519_private_key(&reader.next().read_bytes()?),
#[cfg(feature = "openssl")]
KeyType::RSA => {
let seq = &reader.next().read_bytes()?;
Expand Down
32 changes: 32 additions & 0 deletions russh-keys/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -550,6 +550,38 @@ QR+u0AypRPmzHnOPAAAAEXJvb3RAMTQwOTExNTQ5NDBkAQ==
decode_secret_key(ED25519_AESCTR_KEY, Some("test")).unwrap();
}

// Key from RFC 8410 Section 10.3. This is a key using PrivateKeyInfo structure.
const RFC8410_ED25519_PRIVATE_ONLY_KEY: &str = "-----BEGIN PRIVATE KEY-----
MC4CAQAwBQYDK2VwBCIEINTuctv5E1hK1bbY8fdp+K06/nwoy/HU++CXqI9EdVhC
-----END PRIVATE KEY-----";

#[test]
fn test_decode_rfc8410_ed25519_private_only_key() {
env_logger::try_init().unwrap_or(());
assert!(matches!(
decode_secret_key(RFC8410_ED25519_PRIVATE_ONLY_KEY, None),
Ok(key::KeyPair::Ed25519 { .. })
));
// We always encode public key, skip test_decode_encode_symmetry.
}

// Key from RFC 8410 Section 10.3. This is a key using OneAsymmetricKey structure.
const RFC8410_ED25519_PRIVATE_PUBLIC_KEY: &str = "-----BEGIN PRIVATE KEY-----
MHICAQEwBQYDK2VwBCIEINTuctv5E1hK1bbY8fdp+K06/nwoy/HU++CXqI9EdVhC
oB8wHQYKKoZIhvcNAQkJFDEPDA1DdXJkbGUgQ2hhaXJzgSEAGb9ECWmEzf6FQbrB
Z9w7lshQhqowtrbLDFw4rXAxZuE=
-----END PRIVATE KEY-----";

#[test]
fn test_decode_rfc8410_ed25519_private_public_key() {
env_logger::try_init().unwrap_or(());
assert!(matches!(
decode_secret_key(RFC8410_ED25519_PRIVATE_PUBLIC_KEY, None),
Ok(key::KeyPair::Ed25519 { .. })
));
// We can't encode attributes, skip test_decode_encode_symmetry.
}

#[test]
#[cfg(feature = "openssl")]
fn test_decode_rsa_secret_key() {
Expand Down

0 comments on commit 120c1a0

Please sign in to comment.