Skip to content

Commit

Permalink
Move PrivateCredential to the credential module
Browse files Browse the repository at this point in the history
Signed-off-by: Wiktor Kwapisiewicz <wiktor@metacode.biz>
  • Loading branch information
wiktor-k committed Jul 31, 2024
1 parent ae7d5a5 commit dcec785
Show file tree
Hide file tree
Showing 3 changed files with 116 additions and 120 deletions.
3 changes: 1 addition & 2 deletions src/proto/message/add_remove.rs
Original file line number Diff line number Diff line change
@@ -1,15 +1,14 @@
//! Add a key to an agent with or without constraints and supporting data types.
mod constrained;
mod credential;

pub use constrained::*;
pub use credential::*;
use secrecy::ExposeSecret as _;
use secrecy::SecretString;
use ssh_encoding::{self, CheckedSum, Decode, Encode, Reader, Writer};
use ssh_key::public::KeyData;

use super::PrivateCredential;
use crate::proto::{Error, Result};

/// Add a key to an agent.
Expand Down
115 changes: 0 additions & 115 deletions src/proto/message/add_remove/credential.rs

This file was deleted.

118 changes: 115 additions & 3 deletions src/proto/message/credential.rs
Original file line number Diff line number Diff line change
@@ -1,7 +1,119 @@
use ssh_encoding::{Decode, Encode, Reader};
use ssh_key::{public::KeyData, Certificate};
//! A container for a public / private key pair, or a certificate / private key.
use crate::proto::Error;
use std::str::FromStr as _;

use ssh_encoding::{self, CheckedSum, Decode, Encode, Reader, Writer};
use ssh_key::public::KeyData;
use ssh_key::{certificate::Certificate, private::KeypairData, Algorithm};

use crate::proto::{Error, PrivateKeyData, Result};

/// A container for a public / private key pair, or a certificate / private key.
///
/// When adding an identity to an agent, a user can provide either:
/// 1. A public / private key pair
/// 2. An OpenSSH [certificate](https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.certkeys)
///
/// This structure covers both types of identities a user may
/// send to an agent as part of a [`Request::AddIdentity`](crate::proto::Request::AddIdentity) message.
#[derive(Clone, PartialEq, Debug)]
pub enum PrivateCredential {
/// A public/private key pair
Key {
/// Public/private key pair data
privkey: KeypairData,

/// Key comment, if any.
comment: String,
},

/// An OpenSSH [certificate](https://github.com/openssh/openssh-portable/blob/master/PROTOCOL.certkeys)
Cert {
/// Certificate algorithm.
algorithm: Algorithm,

/// Certificate data.
certificate: Certificate,

/// Private key data.
privkey: PrivateKeyData,

/// Comment, if any.
comment: String,
},
}

impl Decode for PrivateCredential {
type Error = Error;

fn decode(reader: &mut impl Reader) -> Result<Self> {
let alg = String::decode(reader)?;
let cert_alg = Algorithm::new_certificate(&alg);

if let Ok(algorithm) = cert_alg {
let certificate = reader.read_prefixed(|reader| {
let cert = Certificate::decode(reader)?;
Ok::<_, Error>(cert)
})?;
let privkey = PrivateKeyData::decode_as(reader, algorithm.clone())?;
let comment = String::decode(reader)?;

Ok(PrivateCredential::Cert {
algorithm,
certificate,
privkey,
comment,
})
} else {
let algorithm = Algorithm::from_str(&alg).map_err(ssh_encoding::Error::from)?;
let privkey = KeypairData::decode_as(reader, algorithm)?;
let comment = String::decode(reader)?;
Ok(PrivateCredential::Key { privkey, comment })
}
}
}

impl Encode for PrivateCredential {
fn encoded_len(&self) -> ssh_encoding::Result<usize> {
match self {
Self::Key { privkey, comment } => {
[privkey.encoded_len()?, comment.encoded_len()?].checked_sum()
}
Self::Cert {
algorithm,
certificate,
privkey,
comment,
} => [
algorithm.to_certificate_type().encoded_len()?,
certificate.encoded_len_prefixed()?,
privkey.encoded_len()?,
comment.encoded_len()?,
]
.checked_sum(),
}
}

fn encode(&self, writer: &mut impl Writer) -> ssh_encoding::Result<()> {
match self {
Self::Key { privkey, comment } => {
privkey.encode(writer)?;
comment.encode(writer)
}
Self::Cert {
algorithm,
certificate,
privkey,
comment,
} => {
algorithm.to_certificate_type().encode(writer)?;
certificate.encode_prefixed(writer)?;
privkey.encode(writer)?;
comment.encode(writer)
}
}
}
}

#[derive(Debug, PartialEq, Eq, Clone)]
/// Represents a public credential.
Expand Down

0 comments on commit dcec785

Please sign in to comment.