1use crate::Result;
4
5#[cfg(feature = "alloc")]
6use der::{Document, SecretDocument};
7
8#[cfg(feature = "pem")]
9use {
10 crate::LineEnding,
11 alloc::string::String,
12 der::{pem::PemLabel, zeroize::Zeroizing},
13};
14
15#[cfg(feature = "pkcs8")]
16use {
17 crate::{ALGORITHM_ID, ALGORITHM_OID},
18 der::asn1::BitStringRef,
19};
20
21#[cfg(feature = "std")]
22use std::path::Path;
23
24#[cfg(all(feature = "alloc", feature = "pkcs8"))]
25use der::Decode;
26
27#[cfg(all(feature = "alloc", any(feature = "pem", feature = "pkcs8")))]
28use crate::{RsaPrivateKey, RsaPublicKey};
29
30pub trait DecodeRsaPrivateKey: Sized {
32 fn from_pkcs1_der(bytes: &[u8]) -> Result<Self>;
35
36 #[cfg(feature = "pem")]
44 fn from_pkcs1_pem(s: &str) -> Result<Self> {
45 let (label, doc) = SecretDocument::from_pem(s)?;
46 RsaPrivateKey::validate_pem_label(label)?;
47 Self::from_pkcs1_der(doc.as_bytes())
48 }
49
50 #[cfg(feature = "std")]
53 fn read_pkcs1_der_file(path: impl AsRef<Path>) -> Result<Self> {
54 Self::from_pkcs1_der(SecretDocument::read_der_file(path)?.as_bytes())
55 }
56
57 #[cfg(all(feature = "pem", feature = "std"))]
59 fn read_pkcs1_pem_file(path: impl AsRef<Path>) -> Result<Self> {
60 let (label, doc) = SecretDocument::read_pem_file(path)?;
61 RsaPrivateKey::validate_pem_label(&label)?;
62 Self::from_pkcs1_der(doc.as_bytes())
63 }
64}
65
66pub trait DecodeRsaPublicKey: Sized {
68 fn from_pkcs1_der(bytes: &[u8]) -> Result<Self>;
71
72 #[cfg(feature = "pem")]
80 fn from_pkcs1_pem(s: &str) -> Result<Self> {
81 let (label, doc) = Document::from_pem(s)?;
82 RsaPublicKey::validate_pem_label(label)?;
83 Self::from_pkcs1_der(doc.as_bytes())
84 }
85
86 #[cfg(feature = "std")]
89 fn read_pkcs1_der_file(path: impl AsRef<Path>) -> Result<Self> {
90 let doc = Document::read_der_file(path)?;
91 Self::from_pkcs1_der(doc.as_bytes())
92 }
93
94 #[cfg(all(feature = "pem", feature = "std"))]
96 fn read_pkcs1_pem_file(path: impl AsRef<Path>) -> Result<Self> {
97 let (label, doc) = Document::read_pem_file(path)?;
98 RsaPublicKey::validate_pem_label(&label)?;
99 Self::from_pkcs1_der(doc.as_bytes())
100 }
101}
102
103#[cfg(feature = "alloc")]
105pub trait EncodeRsaPrivateKey {
106 fn to_pkcs1_der(&self) -> Result<SecretDocument>;
108
109 #[cfg(feature = "pem")]
111 fn to_pkcs1_pem(&self, line_ending: LineEnding) -> Result<Zeroizing<String>> {
112 let doc = self.to_pkcs1_der()?;
113 Ok(doc.to_pem(RsaPrivateKey::PEM_LABEL, line_ending)?)
114 }
115
116 #[cfg(feature = "std")]
118 fn write_pkcs1_der_file(&self, path: impl AsRef<Path>) -> Result<()> {
119 Ok(self.to_pkcs1_der()?.write_der_file(path)?)
120 }
121
122 #[cfg(all(feature = "pem", feature = "std"))]
124 fn write_pkcs1_pem_file(&self, path: impl AsRef<Path>, line_ending: LineEnding) -> Result<()> {
125 let doc = self.to_pkcs1_der()?;
126 Ok(doc.write_pem_file(path, RsaPrivateKey::PEM_LABEL, line_ending)?)
127 }
128}
129
130#[cfg(feature = "alloc")]
132pub trait EncodeRsaPublicKey {
133 fn to_pkcs1_der(&self) -> Result<Document>;
135
136 #[cfg(feature = "pem")]
138 fn to_pkcs1_pem(&self, line_ending: LineEnding) -> Result<String> {
139 let doc = self.to_pkcs1_der()?;
140 Ok(doc.to_pem(RsaPublicKey::PEM_LABEL, line_ending)?)
141 }
142
143 #[cfg(feature = "std")]
145 fn write_pkcs1_der_file(&self, path: impl AsRef<Path>) -> Result<()> {
146 Ok(self.to_pkcs1_der()?.write_der_file(path)?)
147 }
148
149 #[cfg(all(feature = "pem", feature = "std"))]
151 fn write_pkcs1_pem_file(&self, path: impl AsRef<Path>, line_ending: LineEnding) -> Result<()> {
152 let doc = self.to_pkcs1_der()?;
153 Ok(doc.write_pem_file(path, RsaPublicKey::PEM_LABEL, line_ending)?)
154 }
155}
156
157#[cfg(feature = "pkcs8")]
158impl<T> DecodeRsaPrivateKey for T
159where
160 T: for<'a> TryFrom<pkcs8::PrivateKeyInfo<'a>, Error = pkcs8::Error>,
161{
162 fn from_pkcs1_der(private_key: &[u8]) -> Result<Self> {
163 Ok(Self::try_from(pkcs8::PrivateKeyInfo {
164 algorithm: ALGORITHM_ID,
165 private_key,
166 public_key: None,
167 })?)
168 }
169}
170
171#[cfg(feature = "pkcs8")]
172impl<T> DecodeRsaPublicKey for T
173where
174 T: for<'a> TryFrom<pkcs8::SubjectPublicKeyInfoRef<'a>, Error = pkcs8::spki::Error>,
175{
176 fn from_pkcs1_der(public_key: &[u8]) -> Result<Self> {
177 Ok(Self::try_from(pkcs8::SubjectPublicKeyInfoRef {
178 algorithm: ALGORITHM_ID,
179 subject_public_key: BitStringRef::from_bytes(public_key)?,
180 })?)
181 }
182}
183
184#[cfg(all(feature = "alloc", feature = "pkcs8"))]
185impl<T: pkcs8::EncodePrivateKey> EncodeRsaPrivateKey for T {
186 fn to_pkcs1_der(&self) -> Result<SecretDocument> {
187 let pkcs8_doc = self.to_pkcs8_der()?;
188 let pkcs8_key = pkcs8::PrivateKeyInfo::from_der(pkcs8_doc.as_bytes())?;
189 pkcs8_key.algorithm.assert_algorithm_oid(ALGORITHM_OID)?;
190 RsaPrivateKey::from_der(pkcs8_key.private_key)?.try_into()
191 }
192}
193
194#[cfg(all(feature = "alloc", feature = "pkcs8"))]
195impl<T: pkcs8::EncodePublicKey> EncodeRsaPublicKey for T {
196 fn to_pkcs1_der(&self) -> Result<Document> {
197 let doc = self.to_public_key_der()?;
198 let spki = pkcs8::SubjectPublicKeyInfoRef::from_der(doc.as_bytes())?;
199 spki.algorithm.assert_algorithm_oid(ALGORITHM_OID)?;
200 RsaPublicKey::from_der(spki.subject_public_key.raw_bytes())?.try_into()
201 }
202}