summaryrefslogtreecommitdiffstats
path: root/vendor/openssl/src/base64.rs
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--vendor/openssl/src/base64.rs128
1 files changed, 128 insertions, 0 deletions
diff --git a/vendor/openssl/src/base64.rs b/vendor/openssl/src/base64.rs
new file mode 100644
index 0000000..bfa8cbc
--- /dev/null
+++ b/vendor/openssl/src/base64.rs
@@ -0,0 +1,128 @@
+//! Base64 encoding support.
+use crate::error::ErrorStack;
+use crate::{cvt_n, LenType};
+use libc::c_int;
+use openssl_macros::corresponds;
+
+/// Encodes a slice of bytes to a base64 string.
+///
+/// # Panics
+///
+/// Panics if the input length or computed output length overflow a signed C integer.
+#[corresponds(EVP_EncodeBlock)]
+pub fn encode_block(src: &[u8]) -> String {
+ assert!(src.len() <= c_int::max_value() as usize);
+ let src_len = src.len() as LenType;
+
+ let len = encoded_len(src_len).unwrap();
+ let mut out = Vec::with_capacity(len as usize);
+
+ // SAFETY: `encoded_len` ensures space for 4 output characters
+ // for every 3 input bytes including padding and nul terminator.
+ // `EVP_EncodeBlock` will write only single byte ASCII characters.
+ // `EVP_EncodeBlock` will only write to not read from `out`.
+ unsafe {
+ let out_len = ffi::EVP_EncodeBlock(out.as_mut_ptr(), src.as_ptr(), src_len);
+ out.set_len(out_len as usize);
+ String::from_utf8_unchecked(out)
+ }
+}
+
+/// Decodes a base64-encoded string to bytes.
+///
+/// # Panics
+///
+/// Panics if the input length or computed output length overflow a signed C integer.
+#[corresponds(EVP_DecodeBlock)]
+pub fn decode_block(src: &str) -> Result<Vec<u8>, ErrorStack> {
+ let src = src.trim();
+
+ // https://github.com/openssl/openssl/issues/12143
+ if src.is_empty() {
+ return Ok(vec![]);
+ }
+
+ assert!(src.len() <= c_int::max_value() as usize);
+ let src_len = src.len() as LenType;
+
+ let len = decoded_len(src_len).unwrap();
+ let mut out = Vec::with_capacity(len as usize);
+
+ // SAFETY: `decoded_len` ensures space for 3 output bytes
+ // for every 4 input characters including padding.
+ // `EVP_DecodeBlock` can write fewer bytes after stripping
+ // leading and trailing whitespace, but never more.
+ // `EVP_DecodeBlock` will only write to not read from `out`.
+ unsafe {
+ let out_len = cvt_n(ffi::EVP_DecodeBlock(
+ out.as_mut_ptr(),
+ src.as_ptr(),
+ src_len,
+ ))?;
+ out.set_len(out_len as usize);
+ }
+
+ if src.ends_with('=') {
+ out.pop();
+ if src.ends_with("==") {
+ out.pop();
+ }
+ }
+
+ Ok(out)
+}
+
+fn encoded_len(src_len: LenType) -> Option<LenType> {
+ let mut len = (src_len / 3).checked_mul(4)?;
+
+ if src_len % 3 != 0 {
+ len = len.checked_add(4)?;
+ }
+
+ len = len.checked_add(1)?;
+
+ Some(len)
+}
+
+fn decoded_len(src_len: LenType) -> Option<LenType> {
+ let mut len = (src_len / 4).checked_mul(3)?;
+
+ if src_len % 4 != 0 {
+ len = len.checked_add(3)?;
+ }
+
+ Some(len)
+}
+
+#[cfg(test)]
+mod tests {
+ use super::*;
+
+ #[test]
+ fn test_encode_block() {
+ assert_eq!("".to_string(), encode_block(b""));
+ assert_eq!("Zg==".to_string(), encode_block(b"f"));
+ assert_eq!("Zm8=".to_string(), encode_block(b"fo"));
+ assert_eq!("Zm9v".to_string(), encode_block(b"foo"));
+ assert_eq!("Zm9vYg==".to_string(), encode_block(b"foob"));
+ assert_eq!("Zm9vYmE=".to_string(), encode_block(b"fooba"));
+ assert_eq!("Zm9vYmFy".to_string(), encode_block(b"foobar"));
+ }
+
+ #[test]
+ fn test_decode_block() {
+ assert_eq!(b"".to_vec(), decode_block("").unwrap());
+ assert_eq!(b"f".to_vec(), decode_block("Zg==").unwrap());
+ assert_eq!(b"fo".to_vec(), decode_block("Zm8=").unwrap());
+ assert_eq!(b"foo".to_vec(), decode_block("Zm9v").unwrap());
+ assert_eq!(b"foob".to_vec(), decode_block("Zm9vYg==").unwrap());
+ assert_eq!(b"fooba".to_vec(), decode_block("Zm9vYmE=").unwrap());
+ assert_eq!(b"foobar".to_vec(), decode_block("Zm9vYmFy").unwrap());
+ }
+
+ #[test]
+ fn test_strip_whitespace() {
+ assert_eq!(b"foobar".to_vec(), decode_block(" Zm9vYmFy\n").unwrap());
+ assert_eq!(b"foob".to_vec(), decode_block(" Zm9vYg==\n").unwrap());
+ }
+}