2017-11-23 16:00:47 +00:00
|
|
|
// Copyright (c) 2017 Emmanuel Gil Peyrot <linkmauve@linkmauve.fr>
|
|
|
|
//
|
|
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
// file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
|
2019-01-13 11:39:51 +00:00
|
|
|
use crate::util::error::Error;
|
2018-12-18 14:32:05 +00:00
|
|
|
use base64;
|
2017-11-23 16:00:47 +00:00
|
|
|
|
2017-11-23 16:19:04 +00:00
|
|
|
/// Codec for plain text content.
|
|
|
|
pub struct PlainText;
|
|
|
|
|
|
|
|
impl PlainText {
|
|
|
|
pub fn decode(s: &str) -> Result<Option<String>, Error> {
|
|
|
|
Ok(match s {
|
|
|
|
"" => None,
|
|
|
|
text => Some(text.to_owned()),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn encode(string: &Option<String>) -> Option<String> {
|
2019-02-21 20:00:58 +00:00
|
|
|
string.as_ref().map(ToOwned::to_owned)
|
2017-11-23 16:19:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-24 04:46:08 +00:00
|
|
|
/// Codec for trimmed plain text content.
|
|
|
|
pub struct TrimmedPlainText;
|
|
|
|
|
|
|
|
impl TrimmedPlainText {
|
|
|
|
pub fn decode(s: &str) -> Result<String, Error> {
|
|
|
|
Ok(match s.trim() {
|
|
|
|
"" => return Err(Error::ParseError("URI missing in uri.")),
|
|
|
|
text => text.to_owned(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:00:58 +00:00
|
|
|
pub fn encode(string: &str) -> String {
|
2017-11-24 04:46:08 +00:00
|
|
|
string.to_owned()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-27 16:16:23 +00:00
|
|
|
/// Codec wrapping base64 encode/decode.
|
2017-11-23 16:00:47 +00:00
|
|
|
pub struct Base64;
|
|
|
|
|
|
|
|
impl Base64 {
|
|
|
|
pub fn decode(s: &str) -> Result<Vec<u8>, Error> {
|
|
|
|
Ok(base64::decode(s)?)
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:00:58 +00:00
|
|
|
pub fn encode(b: &[u8]) -> Option<String> {
|
2017-11-23 16:19:04 +00:00
|
|
|
Some(base64::encode(b))
|
2017-11-23 16:00:47 +00:00
|
|
|
}
|
|
|
|
}
|
2019-01-27 16:16:23 +00:00
|
|
|
|
|
|
|
/// Codec wrapping base64 encode/decode, while ignoring whitespace characters.
|
|
|
|
pub struct WhitespaceAwareBase64;
|
|
|
|
|
|
|
|
impl WhitespaceAwareBase64 {
|
|
|
|
pub fn decode(s: &str) -> Result<Vec<u8>, Error> {
|
2019-02-21 20:00:58 +00:00
|
|
|
let s: String = s.chars().filter(|ch| *ch != ' ' && *ch != '\n' && *ch != '\t').collect();
|
2019-01-27 16:16:23 +00:00
|
|
|
Ok(base64::decode(&s)?)
|
|
|
|
}
|
|
|
|
|
2019-02-21 20:00:58 +00:00
|
|
|
pub fn encode(b: &[u8]) -> Option<String> {
|
2019-01-27 16:16:23 +00:00
|
|
|
Some(base64::encode(b))
|
|
|
|
}
|
|
|
|
}
|
2019-02-28 12:32:52 +00:00
|
|
|
|
|
|
|
/// Codec for colon-separated bytes of uppercase hexadecimal.
|
|
|
|
pub struct ColonSeparatedHex;
|
|
|
|
|
|
|
|
impl ColonSeparatedHex {
|
|
|
|
pub fn decode(s: &str) -> Result<Vec<u8>, Error> {
|
|
|
|
let mut bytes = vec![];
|
|
|
|
for i in 0..(1 + s.len()) / 3 {
|
|
|
|
let byte = u8::from_str_radix(&s[3 * i..3 * i + 2], 16)?;
|
|
|
|
if 3 * i + 2 < s.len() {
|
|
|
|
assert_eq!(&s[3 * i + 2..3 * i + 3], ":");
|
|
|
|
}
|
|
|
|
bytes.push(byte);
|
|
|
|
}
|
|
|
|
Ok(bytes)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn encode(b: &[u8]) -> Option<String> {
|
|
|
|
let mut bytes = vec![];
|
|
|
|
for byte in b {
|
|
|
|
bytes.push(format!("{:02X}", byte));
|
|
|
|
}
|
|
|
|
Some(bytes.join(":"))
|
|
|
|
}
|
|
|
|
}
|