2021-10-24 02:25:42 -07:00
|
|
|
use std::fmt::{Display, Formatter};
|
|
|
|
use std::sync::Arc;
|
2021-10-23 10:10:55 -07:00
|
|
|
|
|
|
|
use gloo_console::log;
|
|
|
|
use image::GenericImageView;
|
|
|
|
use js_sys::{Array, Uint8Array};
|
2021-10-24 02:25:42 -07:00
|
|
|
use omegaupload_common::crypto::{open_in_place, Key, Nonce};
|
2021-10-23 10:10:55 -07:00
|
|
|
use wasm_bindgen::JsCast;
|
|
|
|
use web_sys::Blob;
|
|
|
|
|
2021-10-24 11:40:19 -07:00
|
|
|
#[derive(Clone)]
|
|
|
|
pub enum DecryptedData {
|
|
|
|
String(Arc<String>),
|
|
|
|
Blob(Arc<Blob>),
|
|
|
|
Image(Arc<Blob>, (u32, u32), usize),
|
|
|
|
Audio(Arc<Blob>),
|
|
|
|
Video(Arc<Blob>),
|
|
|
|
}
|
2021-10-23 10:10:55 -07:00
|
|
|
|
2021-10-24 02:25:42 -07:00
|
|
|
pub fn decrypt(
|
2021-10-23 10:10:55 -07:00
|
|
|
mut container: Vec<u8>,
|
|
|
|
key: Key,
|
|
|
|
nonce: Nonce,
|
|
|
|
maybe_password: Option<Key>,
|
|
|
|
) -> Result<DecryptedData, PasteCompleteConstructionError> {
|
|
|
|
let container = &mut container;
|
|
|
|
log!("stage 1 decryption start");
|
|
|
|
if let Some(password) = maybe_password {
|
|
|
|
open_in_place(container, &nonce.increment(), &password)
|
|
|
|
.map_err(|_| PasteCompleteConstructionError::StageOneFailure)?;
|
|
|
|
}
|
|
|
|
|
|
|
|
log!("stage 2 decryption start");
|
|
|
|
open_in_place(container, &nonce, &key)
|
|
|
|
.map_err(|_| PasteCompleteConstructionError::StageTwoFailure)?;
|
|
|
|
|
|
|
|
log!("stage 2 decryption end");
|
2021-10-24 12:14:55 -07:00
|
|
|
if let Ok(decrypted) = std::str::from_utf8(container) {
|
2021-10-23 10:10:55 -07:00
|
|
|
Ok(DecryptedData::String(Arc::new(decrypted.to_owned())))
|
|
|
|
} else {
|
|
|
|
log!("blob conversion start");
|
|
|
|
let blob_chunks = Array::new_with_length(container.chunks(65536).len().try_into().unwrap());
|
|
|
|
for (i, chunk) in container.chunks(65536).enumerate() {
|
|
|
|
let array = Uint8Array::new_with_length(chunk.len().try_into().unwrap());
|
2021-10-24 12:14:55 -07:00
|
|
|
array.copy_from(chunk);
|
2021-10-23 10:10:55 -07:00
|
|
|
blob_chunks.set(i.try_into().unwrap(), array.dyn_into().unwrap());
|
|
|
|
}
|
|
|
|
let blob =
|
|
|
|
Arc::new(Blob::new_with_u8_array_sequence(blob_chunks.dyn_ref().unwrap()).unwrap());
|
|
|
|
log!("blob conversion end");
|
|
|
|
|
2021-10-24 12:14:55 -07:00
|
|
|
if let Ok(image) = image::load_from_memory(container) {
|
2021-10-23 10:10:55 -07:00
|
|
|
Ok(DecryptedData::Image(
|
|
|
|
blob,
|
|
|
|
image.dimensions(),
|
|
|
|
container.len(),
|
|
|
|
))
|
|
|
|
} else {
|
2021-10-24 12:14:55 -07:00
|
|
|
let mime_type = tree_magic_mini::from_u8(container);
|
2021-10-24 11:40:19 -07:00
|
|
|
log!(mime_type);
|
|
|
|
if mime_type.starts_with("audio/") {
|
|
|
|
Ok(DecryptedData::Audio(blob))
|
|
|
|
} else if mime_type.starts_with("video/") || mime_type.ends_with("x-matroska") {
|
|
|
|
Ok(DecryptedData::Video(blob))
|
|
|
|
} else {
|
|
|
|
Ok(DecryptedData::Blob(blob))
|
|
|
|
}
|
2021-10-23 10:10:55 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-10-24 02:25:42 -07:00
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub enum PasteCompleteConstructionError {
|
|
|
|
StageOneFailure,
|
|
|
|
StageTwoFailure,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl std::error::Error for PasteCompleteConstructionError {}
|
|
|
|
|
|
|
|
impl Display for PasteCompleteConstructionError {
|
|
|
|
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
|
|
|
|
match self {
|
|
|
|
PasteCompleteConstructionError::StageOneFailure => {
|
|
|
|
write!(f, "Failed to decrypt stage one.")
|
|
|
|
}
|
|
|
|
PasteCompleteConstructionError::StageTwoFailure => {
|
|
|
|
write!(f, "Failed to decrypt stage two.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|