#![deny(missing_docs,
missing_debug_implementations, missing_copy_implementations,
trivial_casts, trivial_numeric_casts,
unsafe_code,
unstable_features,
unused_import_braces, unused_qualifications)]
extern crate rustc_serialize;
extern crate regex;
use rustc_serialize::base64::FromBase64;
use regex::{Captures, Regex};
const PEM_SECTION: &'static str =
r"(?s)-----BEGIN (?P<begin>.*?)-----\s*(?P<data>.*?)-----END (?P<end>.*?)-----\s*";
#[derive(PartialEq,Clone,Copy,Debug)]
pub enum Error {
PemFraming,
InvalidBeginTag,
InvalidEndTag,
MismatchedTags,
InvalidData,
#[doc(hidden)]
__Nonexhaustive,
}
#[derive(Debug)]
pub struct Pem {
pub tag: String,
pub contents: Vec<u8>,
}
fn parse_helper(caps: Captures) -> Result<Pem, Error> {
let tag = caps.name("begin").unwrap();
if tag == "" {
return Err(Error::InvalidBeginTag);
}
let tag_end = caps.name("end").unwrap();
if tag_end == "" {
return Err(Error::InvalidEndTag);
}
if tag != tag_end {
return Err(Error::MismatchedTags);
}
let data = caps.name("data").unwrap();
let data = data.replace("\n", "").replace(" ", "");
let contents = match data.from_base64() {
Ok(c) => c,
Err(_) => {
return Err(Error::InvalidData);
}
};
Ok(Pem {
tag: tag.to_owned(),
contents: contents,
})
}
pub fn parse(input: &str) -> Result<Pem, Error> {
let re = Regex::new(PEM_SECTION).unwrap();
match re.captures(input) {
Some(caps) => parse_helper(caps),
None => Err(Error::PemFraming),
}
}
pub fn parse_many(input: &str) -> Vec<Pem> {
let re = Regex::new(PEM_SECTION).unwrap();
re.captures_iter(input)
.filter_map(|caps| parse_helper(caps).ok())
.collect()
}
#[cfg(test)]
mod test {
const SAMPLE: &'static str = "-----BEGIN RSA PRIVATE KEY-----
MIIBPQIBAAJBAOsfi5AGYhdRs/x6q5H7kScxA0Kzzqe6WI6gf6+tc6IvKQJo5rQc
dWWSQ0nRGt2hOPDO+35NKhQEjBQxPh/v7n0CAwEAAQJBAOGaBAyuw0ICyENy5NsO
2gkT00AWTSzM9Zns0HedY31yEabkuFvrMCHjscEF7u3Y6PB7An3IzooBHchsFDei
AAECIQD/JahddzR5K3A6rzTidmAf1PBtqi7296EnWv8WvpfAAQIhAOvowIXZI4Un
DXjgZ9ekuUjZN+GUQRAVlkEEohGLVy59AiEA90VtqDdQuWWpvJX0cM08V10tLXrT
TTGsEtITid1ogAECIQDAaFl90ZgS5cMrL3wCeatVKzVUmuJmB/VAmlLFFGzK0QIh
ANJGc7AFk4fyFD/OezhwGHbWmo/S+bfeAiIh2Ss2FxKJ
-----END RSA PRIVATE KEY-----
-----BEGIN RSA PUBLIC KEY-----
MIIBOgIBAAJBAMIeCnn9G/7g2Z6J+qHOE2XCLLuPoh5NHTO2Fm+PbzBvafBo0oYo
QVVy7frzxmOqx6iIZBxTyfAQqBPO3Br59BMCAwEAAQJAX+PjHPuxdqiwF6blTkS0
RFI1MrnzRbCmOkM6tgVO0cd6r5Z4bDGLusH9yjI9iI84gPRjK0AzymXFmBGuREHI
sQIhAPKf4pp+Prvutgq2ayygleZChBr1DC4XnnufBNtaswyvAiEAzNGVKgNvzuhk
ijoUXIDruJQEGFGvZTsi1D2RehXiT90CIQC4HOQUYKCydB7oWi1SHDokFW2yFyo6
/+lf3fgNjPI6OQIgUPmTFXciXxT1msh3gFLf3qt2Kv8wbr9Ad9SXjULVpGkCIB+g
RzHX0lkJl9Stshd/7Gbt65/QYq+v+xvAeT0CoyIg
-----END RSA PUBLIC KEY-----
";
#[test]
fn parse_works() {
let pem = super::parse(SAMPLE).unwrap();
assert_eq!(pem.tag, "RSA PRIVATE KEY");
}
#[test]
fn parse_invalid_framing() {
let input = "--BEGIN data-----
-----END data-----";
match super::parse(&input) {
Ok(_) => assert!(false),
Err(code) => assert_eq!(code, super::Error::PemFraming),
}
}
#[test]
fn parse_invalid_begin() {
let input = "-----BEGIN -----
MIIBOgIBAAJBAMIeCnn9G/7g2Z6J+qHOE2XCLLuPoh5NHTO2Fm+PbzBvafBo0oYo
QVVy7frzxmOqx6iIZBxTyfAQqBPO3Br59BMCAwEAAQJAX+PjHPuxdqiwF6blTkS0
RFI1MrnzRbCmOkM6tgVO0cd6r5Z4bDGLusH9yjI9iI84gPRjK0AzymXFmBGuREHI
sQIhAPKf4pp+Prvutgq2ayygleZChBr1DC4XnnufBNtaswyvAiEAzNGVKgNvzuhk
ijoUXIDruJQEGFGvZTsi1D2RehXiT90CIQC4HOQUYKCydB7oWi1SHDokFW2yFyo6
/+lf3fgNjPI6OQIgUPmTFXciXxT1msh3gFLf3qt2Kv8wbr9Ad9SXjULVpGkCIB+g
RzHX0lkJl9Stshd/7Gbt65/QYq+v+xvAeT0CoyIg
-----END RSA PUBLIC KEY-----";
match super::parse(&input) {
Ok(_) => assert!(false),
Err(code) => assert_eq!(code, super::Error::InvalidBeginTag),
}
}
#[test]
fn parse_invalid_end() {
let input = "-----BEGIN DATA-----
MIIBOgIBAAJBAMIeCnn9G/7g2Z6J+qHOE2XCLLuPoh5NHTO2Fm+PbzBvafBo0oYo
QVVy7frzxmOqx6iIZBxTyfAQqBPO3Br59BMCAwEAAQJAX+PjHPuxdqiwF6blTkS0
RFI1MrnzRbCmOkM6tgVO0cd6r5Z4bDGLusH9yjI9iI84gPRjK0AzymXFmBGuREHI
sQIhAPKf4pp+Prvutgq2ayygleZChBr1DC4XnnufBNtaswyvAiEAzNGVKgNvzuhk
ijoUXIDruJQEGFGvZTsi1D2RehXiT90CIQC4HOQUYKCydB7oWi1SHDokFW2yFyo6
/+lf3fgNjPI6OQIgUPmTFXciXxT1msh3gFLf3qt2Kv8wbr9Ad9SXjULVpGkCIB+g
RzHX0lkJl9Stshd/7Gbt65/QYq+v+xvAeT0CoyIg
-----END -----";
match super::parse(&input) {
Ok(_) => assert!(false),
Err(code) => assert_eq!(code, super::Error::InvalidEndTag),
}
}
#[test]
fn parse_invalid_data() {
let input = "-----BEGIN DATA-----
MIIBOgIBAAJBAMIeCnn9G/7g2Z6J+qHOE2XCLLuPoh5NHTO2Fm+PbzBvafBo0oY?
QVVy7frzxmOqx6iIZBxTyfAQqBPO3Br59BMCAwEAAQJAX+PjHPuxdqiwF6blTkS0
RFI1MrnzRbCmOkM6tgVO0cd6r5Z4bDGLusH9yjI9iI84gPRjK0AzymXFmBGuREHI
sQIhAPKf4pp+Prvutgq2ayygleZChBr1DC4XnnufBNtaswyvAiEAzNGVKgNvzuhk
ijoUXIDruJQEGFGvZTsi1D2RehXiT90CIQC4HOQUYKCydB7oWi1SHDokFW2yFyo6
/+lf3fgNjPI6OQIgUPmTFXciXxT1msh3gFLf3qt2Kv8wbr9Ad9SXjULVpGkCIB+g
RzHX0lkJl9Stshd/7Gbt65/QYq+v+xvAeT0CoyIg
-----END DATA-----";
match super::parse(&input) {
Ok(_) => assert!(false),
Err(code) => assert_eq!(code, super::Error::InvalidData),
}
}
#[test]
fn parse_empty_data() {
let input = "-----BEGIN DATA-----
-----END DATA-----";
let pem = super::parse(&input).unwrap();
assert_eq!(pem.contents.len(), 0);
}
#[test]
fn parse_many_works() {
let pems = super::parse_many(SAMPLE);
assert_eq!(pems.len(), 2);
assert_eq!(pems[0].tag, "RSA PRIVATE KEY");
assert_eq!(pems[1].tag, "RSA PUBLIC KEY");
}
}