2021-04-06 06:55:05 -04:00
|
|
|
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
2021-04-07 09:22:14 -04:00
|
|
|
use deno_core::error::null_opbuf;
|
|
|
|
use deno_core::error::AnyError;
|
|
|
|
use deno_core::url::Url;
|
2021-04-06 06:55:05 -04:00
|
|
|
use deno_core::JsRuntime;
|
2021-04-07 09:22:14 -04:00
|
|
|
use deno_core::ModuleSpecifier;
|
|
|
|
use deno_core::ZeroCopyBuf;
|
|
|
|
use std::collections::HashMap;
|
2021-04-06 06:55:05 -04:00
|
|
|
use std::path::PathBuf;
|
2021-04-07 09:22:14 -04:00
|
|
|
use std::sync::Arc;
|
|
|
|
use std::sync::Mutex;
|
|
|
|
use uuid::Uuid;
|
|
|
|
|
2021-04-08 21:27:27 -04:00
|
|
|
#[derive(Debug, Clone)]
|
2021-04-07 09:22:14 -04:00
|
|
|
pub struct Blob {
|
|
|
|
pub data: Vec<u8>,
|
|
|
|
pub media_type: String,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Location(pub Url);
|
|
|
|
|
2021-04-08 21:27:27 -04:00
|
|
|
#[derive(Debug, Default, Clone)]
|
2021-04-07 09:22:14 -04:00
|
|
|
pub struct BlobUrlStore(Arc<Mutex<HashMap<Url, Blob>>>);
|
|
|
|
|
|
|
|
impl BlobUrlStore {
|
|
|
|
pub fn get(&self, url: &ModuleSpecifier) -> Result<Option<Blob>, AnyError> {
|
|
|
|
let blob_store = self.0.lock().unwrap();
|
|
|
|
Ok(blob_store.get(url).cloned())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn insert(&self, blob: Blob, maybe_location: Option<Url>) -> Url {
|
|
|
|
let origin = if let Some(location) = maybe_location {
|
|
|
|
location.origin().ascii_serialization()
|
|
|
|
} else {
|
|
|
|
"null".to_string()
|
|
|
|
};
|
|
|
|
let id = Uuid::new_v4();
|
|
|
|
let url = Url::parse(&format!("blob:{}/{}", origin, id)).unwrap();
|
|
|
|
|
|
|
|
let mut blob_store = self.0.lock().unwrap();
|
|
|
|
blob_store.insert(url.clone(), blob);
|
|
|
|
|
|
|
|
url
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn remove(&self, url: &ModuleSpecifier) {
|
|
|
|
let mut blob_store = self.0.lock().unwrap();
|
|
|
|
blob_store.remove(&url);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn op_file_create_object_url(
|
|
|
|
state: &mut deno_core::OpState,
|
|
|
|
media_type: String,
|
|
|
|
zero_copy: Option<ZeroCopyBuf>,
|
|
|
|
) -> Result<String, AnyError> {
|
|
|
|
let data = zero_copy.ok_or_else(null_opbuf)?;
|
|
|
|
let blob = Blob {
|
|
|
|
data: data.to_vec(),
|
|
|
|
media_type,
|
|
|
|
};
|
|
|
|
|
|
|
|
let maybe_location = state.try_borrow::<Location>();
|
|
|
|
let blob_store = state.borrow::<BlobUrlStore>();
|
|
|
|
|
|
|
|
let url =
|
|
|
|
blob_store.insert(blob, maybe_location.map(|location| location.0.clone()));
|
|
|
|
|
|
|
|
Ok(url.to_string())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn op_file_revoke_object_url(
|
|
|
|
state: &mut deno_core::OpState,
|
|
|
|
url: String,
|
|
|
|
_zero_copy: Option<ZeroCopyBuf>,
|
|
|
|
) -> Result<(), AnyError> {
|
|
|
|
let url = Url::parse(&url)?;
|
|
|
|
let blob_store = state.borrow::<BlobUrlStore>();
|
|
|
|
blob_store.remove(&url);
|
|
|
|
Ok(())
|
|
|
|
}
|
2021-04-06 06:55:05 -04:00
|
|
|
|
|
|
|
/// Load and execute the javascript code.
|
|
|
|
pub fn init(isolate: &mut JsRuntime) {
|
|
|
|
let files = vec![
|
|
|
|
("deno:op_crates/file/01_file.js", include_str!("01_file.js")),
|
|
|
|
(
|
|
|
|
"deno:op_crates/file/02_filereader.js",
|
|
|
|
include_str!("02_filereader.js"),
|
|
|
|
),
|
2021-04-07 09:22:14 -04:00
|
|
|
(
|
|
|
|
"deno:op_crates/file/03_blob_url.js",
|
|
|
|
include_str!("03_blob_url.js"),
|
|
|
|
),
|
2021-04-06 06:55:05 -04:00
|
|
|
];
|
|
|
|
for (url, source_code) in files {
|
|
|
|
isolate.execute(url, source_code).unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn get_declaration() -> PathBuf {
|
|
|
|
PathBuf::from(env!("CARGO_MANIFEST_DIR")).join("lib.deno_file.d.ts")
|
|
|
|
}
|