use crate::{Error, PrivateKeyInfoRef, Result};
#[cfg(feature = "alloc")]
use der::SecretDocument;
#[cfg(feature = "encryption")]
use crate::EncryptedPrivateKeyInfoRef;
#[cfg(feature = "pem")]
use {
crate::LineEnding,
alloc::string::String,
der::{
pem::{self, PemLabel},
zeroize::Zeroizing,
},
};
#[cfg(feature = "std")]
use std::path::Path;
pub trait DecodePrivateKey: Sized {
fn from_pkcs8_der(bytes: &[u8]) -> Result<Self>;
#[cfg(feature = "encryption")]
fn from_pkcs8_encrypted_der(bytes: &[u8], password: impl AsRef<[u8]>) -> Result<Self> {
let doc = EncryptedPrivateKeyInfoRef::try_from(bytes)?.decrypt(password)?;
Self::from_pkcs8_der(doc.as_bytes())
}
#[cfg(feature = "pem")]
fn from_pkcs8_pem(s: &str) -> Result<Self> {
let label = pem::decode_label(s.as_bytes())?;
PrivateKeyInfoRef::validate_pem_label(label)?;
let doc = SecretDocument::from_pem(s)?.1;
Self::from_pkcs8_der(doc.as_bytes())
}
#[cfg(all(feature = "encryption", feature = "pem"))]
fn from_pkcs8_encrypted_pem(s: &str, password: impl AsRef<[u8]>) -> Result<Self> {
let (label, doc) = SecretDocument::from_pem(s)?;
EncryptedPrivateKeyInfoRef::validate_pem_label(label)?;
Self::from_pkcs8_encrypted_der(doc.as_bytes(), password)
}
#[cfg(feature = "std")]
fn read_pkcs8_der_file(path: impl AsRef<Path>) -> Result<Self> {
Self::from_pkcs8_der(SecretDocument::read_der_file(path)?.as_bytes())
}
#[cfg(all(feature = "pem", feature = "std"))]
fn read_pkcs8_pem_file(path: impl AsRef<Path>) -> Result<Self> {
let (label, doc) = SecretDocument::read_pem_file(path)?;
PrivateKeyInfoRef::validate_pem_label(&label)?;
Self::from_pkcs8_der(doc.as_bytes())
}
}
impl<T> DecodePrivateKey for T
where
T: for<'a> TryFrom<PrivateKeyInfoRef<'a>, Error = Error>,
{
fn from_pkcs8_der(bytes: &[u8]) -> Result<Self> {
Self::try_from(PrivateKeyInfoRef::try_from(bytes)?)
}
}
#[cfg(feature = "alloc")]
pub trait EncodePrivateKey {
fn to_pkcs8_der(&self) -> Result<SecretDocument>;
#[cfg(feature = "getrandom")]
fn to_pkcs8_encrypted_der(&self, password: impl AsRef<[u8]>) -> Result<SecretDocument> {
EncryptedPrivateKeyInfoRef::encrypt(password, self.to_pkcs8_der()?.as_bytes())
}
#[cfg(feature = "pem")]
fn to_pkcs8_pem(&self, line_ending: LineEnding) -> Result<Zeroizing<String>> {
let doc = self.to_pkcs8_der()?;
Ok(doc.to_pem(PrivateKeyInfoRef::PEM_LABEL, line_ending)?)
}
#[cfg(all(feature = "getrandom", feature = "pem"))]
fn to_pkcs8_encrypted_pem(
&self,
password: impl AsRef<[u8]>,
line_ending: LineEnding,
) -> Result<Zeroizing<String>> {
let doc = self.to_pkcs8_encrypted_der(password)?;
Ok(doc.to_pem(EncryptedPrivateKeyInfoRef::PEM_LABEL, line_ending)?)
}
#[cfg(feature = "std")]
fn write_pkcs8_der_file(&self, path: impl AsRef<Path>) -> Result<()> {
Ok(self.to_pkcs8_der()?.write_der_file(path)?)
}
#[cfg(all(feature = "pem", feature = "std"))]
fn write_pkcs8_pem_file(&self, path: impl AsRef<Path>, line_ending: LineEnding) -> Result<()> {
let doc = self.to_pkcs8_der()?;
Ok(doc.write_pem_file(path, PrivateKeyInfoRef::PEM_LABEL, line_ending)?)
}
}