use mime_guess::Mime;
use serde_json;
use crate::crypto::b64;
const MIME_TAR: &str = "application/x-tar";
#[derive(Debug, Serialize, Deserialize)]
#[serde(untagged)]
pub enum Metadata {
V2 {
name: String,
iv: String,
#[serde(rename = "type")]
mime: String,
},
V3 {
name: String,
#[serde(rename = "type")]
mime: String,
size: u64,
manifest: Manifest,
},
}
impl Metadata {
pub fn from_send2(iv: &[u8], name: String, mime: &Mime) -> Self {
Metadata::V2 {
iv: b64::encode(iv),
name,
mime: mime.to_string(),
}
}
pub fn from_send3(name: String, mime: String, size: u64) -> Self {
Metadata::V3 {
name: name.clone(),
mime: mime.clone(),
size,
manifest: Manifest::from_file(name, mime, size),
}
}
pub fn to_json(&self) -> String {
serde_json::to_string(&self).unwrap()
}
pub fn name(&self) -> &str {
match self {
Metadata::V2 { name, .. } => &name,
Metadata::V3 { name, .. } => &name,
}
}
pub fn mime(&self) -> &str {
match self {
Metadata::V2 { mime, .. } => &mime,
Metadata::V3 { mime, .. } => &mime,
}
}
pub fn iv(&self) -> Option<[u8; 12]> {
let iv = match self {
Metadata::V2 { iv, .. } => iv,
Metadata::V3 { .. } => return None,
};
let decoded = b64::decode(iv).unwrap();
Some(*array_ref!(decoded, 0, 12))
}
pub fn size(&self) -> Option<u64> {
match self {
Metadata::V2 { .. } => None,
Metadata::V3 { size, .. } => Some(*size),
}
}
pub fn is_archive(&self) -> bool {
self.mime().to_lowercase() == MIME_TAR.to_lowercase()
}
}
#[derive(Debug, Serialize, Deserialize)]
pub struct Manifest {
files: Vec<ManifestFile>,
}
impl Manifest {
pub fn from(files: Vec<ManifestFile>) -> Self {
Self { files }
}
pub fn from_file(name: String, mime: String, size: u64) -> Self {
Self::from(vec![ManifestFile::from(name, mime, size)])
}
}
#[derive(Debug, Serialize, Deserialize)]
pub struct ManifestFile {
name: String,
#[serde(rename = "type")]
mime: String,
size: u64,
}
impl ManifestFile {
pub fn from(name: String, mime: String, size: u64) -> Self {
ManifestFile { name, mime, size }
}
}