mirror of
https://github.com/denoland/deno.git
synced 2024-11-22 15:06:54 -05:00
Permissions refactor (#1864)
Refactored permissions to be assignable on a per-isolate basis, and added a fix for #1858 to op_fetch_module_meta_data.
This commit is contained in:
parent
1cc02a5d9d
commit
8c310d3d56
6 changed files with 313 additions and 124 deletions
|
@ -2,6 +2,7 @@
|
||||||
use crate::isolate::Buf;
|
use crate::isolate::Buf;
|
||||||
use crate::isolate::IsolateState;
|
use crate::isolate::IsolateState;
|
||||||
use crate::msg;
|
use crate::msg;
|
||||||
|
use crate::permissions::DenoPermissions;
|
||||||
use crate::resources;
|
use crate::resources;
|
||||||
use crate::resources::Resource;
|
use crate::resources::Resource;
|
||||||
use crate::resources::ResourceId;
|
use crate::resources::ResourceId;
|
||||||
|
@ -10,6 +11,7 @@ use crate::workers;
|
||||||
use futures::Future;
|
use futures::Future;
|
||||||
use serde_json;
|
use serde_json;
|
||||||
use std::str;
|
use std::str;
|
||||||
|
use std::sync::atomic::AtomicBool;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
|
|
||||||
|
@ -48,9 +50,19 @@ impl ModuleMetaData {
|
||||||
|
|
||||||
fn lazy_start(parent_state: &Arc<IsolateState>) -> Resource {
|
fn lazy_start(parent_state: &Arc<IsolateState>) -> Resource {
|
||||||
let mut cell = C_RID.lock().unwrap();
|
let mut cell = C_RID.lock().unwrap();
|
||||||
|
let permissions = DenoPermissions {
|
||||||
|
allow_read: AtomicBool::new(true),
|
||||||
|
allow_write: AtomicBool::new(false),
|
||||||
|
allow_env: AtomicBool::new(false),
|
||||||
|
allow_net: AtomicBool::new(true),
|
||||||
|
allow_run: AtomicBool::new(false),
|
||||||
|
};
|
||||||
let rid = cell.get_or_insert_with(|| {
|
let rid = cell.get_or_insert_with(|| {
|
||||||
let resource =
|
let resource = workers::spawn(
|
||||||
workers::spawn(parent_state.clone(), "compilerMain()".to_string());
|
parent_state.clone(),
|
||||||
|
"compilerMain()".to_string(),
|
||||||
|
permissions,
|
||||||
|
);
|
||||||
resource.rid
|
resource.rid
|
||||||
});
|
});
|
||||||
Resource { rid: *rid }
|
Resource { rid: *rid }
|
||||||
|
|
|
@ -64,6 +64,7 @@ pub struct Isolate {
|
||||||
timeout_due: Cell<Option<Instant>>,
|
timeout_due: Cell<Option<Instant>>,
|
||||||
pub modules: RefCell<Modules>,
|
pub modules: RefCell<Modules>,
|
||||||
pub state: Arc<IsolateState>,
|
pub state: Arc<IsolateState>,
|
||||||
|
pub permissions: Arc<DenoPermissions>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub type WorkerSender = async_mpsc::Sender<Buf>;
|
pub type WorkerSender = async_mpsc::Sender<Buf>;
|
||||||
|
@ -78,7 +79,6 @@ pub type WorkerChannels = (WorkerSender, WorkerReceiver);
|
||||||
pub struct IsolateState {
|
pub struct IsolateState {
|
||||||
pub dir: deno_dir::DenoDir,
|
pub dir: deno_dir::DenoDir,
|
||||||
pub argv: Vec<String>,
|
pub argv: Vec<String>,
|
||||||
pub permissions: DenoPermissions,
|
|
||||||
pub flags: flags::DenoFlags,
|
pub flags: flags::DenoFlags,
|
||||||
pub metrics: Metrics,
|
pub metrics: Metrics,
|
||||||
pub worker_channels: Option<Mutex<WorkerChannels>>,
|
pub worker_channels: Option<Mutex<WorkerChannels>>,
|
||||||
|
@ -96,7 +96,6 @@ impl IsolateState {
|
||||||
dir: deno_dir::DenoDir::new(flags.reload, flags.recompile, custom_root)
|
dir: deno_dir::DenoDir::new(flags.reload, flags.recompile, custom_root)
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
argv: argv_rest,
|
argv: argv_rest,
|
||||||
permissions: DenoPermissions::new(&flags),
|
|
||||||
flags,
|
flags,
|
||||||
metrics: Metrics::default(),
|
metrics: Metrics::default(),
|
||||||
worker_channels: worker_channels.map(Mutex::new),
|
worker_channels: worker_channels.map(Mutex::new),
|
||||||
|
@ -127,31 +126,6 @@ impl IsolateState {
|
||||||
Arc::new(IsolateState::new(flags, rest_argv, None))
|
Arc::new(IsolateState::new(flags, rest_argv, None))
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn check_read(&self, filename: &str) -> DenoResult<()> {
|
|
||||||
self.permissions.check_read(filename)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn check_write(&self, filename: &str) -> DenoResult<()> {
|
|
||||||
self.permissions.check_write(filename)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn check_env(&self) -> DenoResult<()> {
|
|
||||||
self.permissions.check_env()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn check_net(&self, filename: &str) -> DenoResult<()> {
|
|
||||||
self.permissions.check_net(filename)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn check_run(&self) -> DenoResult<()> {
|
|
||||||
self.permissions.check_run()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn metrics_op_dispatched(
|
fn metrics_op_dispatched(
|
||||||
&self,
|
&self,
|
||||||
bytes_sent_control: usize,
|
bytes_sent_control: usize,
|
||||||
|
@ -195,6 +169,7 @@ impl Isolate {
|
||||||
snapshot: libdeno::deno_buf,
|
snapshot: libdeno::deno_buf,
|
||||||
state: Arc<IsolateState>,
|
state: Arc<IsolateState>,
|
||||||
dispatch: Dispatch,
|
dispatch: Dispatch,
|
||||||
|
permissions: DenoPermissions,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
DENO_INIT.call_once(|| {
|
DENO_INIT.call_once(|| {
|
||||||
unsafe { libdeno::deno_init() };
|
unsafe { libdeno::deno_init() };
|
||||||
|
@ -218,6 +193,7 @@ impl Isolate {
|
||||||
timeout_due: Cell::new(None),
|
timeout_due: Cell::new(None),
|
||||||
modules: RefCell::new(Modules::new()),
|
modules: RefCell::new(Modules::new()),
|
||||||
state,
|
state,
|
||||||
|
permissions: Arc::new(permissions),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -242,6 +218,31 @@ impl Isolate {
|
||||||
self.timeout_due.set(inst);
|
self.timeout_due.set(inst);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn check_read(&self, filename: &str) -> DenoResult<()> {
|
||||||
|
self.permissions.check_read(filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn check_write(&self, filename: &str) -> DenoResult<()> {
|
||||||
|
self.permissions.check_write(filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn check_env(&self) -> DenoResult<()> {
|
||||||
|
self.permissions.check_env()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn check_net(&self, filename: &str) -> DenoResult<()> {
|
||||||
|
self.permissions.check_net(filename)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn check_run(&self) -> DenoResult<()> {
|
||||||
|
self.permissions.check_run()
|
||||||
|
}
|
||||||
|
|
||||||
pub fn last_exception(&self) -> Option<JSError> {
|
pub fn last_exception(&self) -> Option<JSError> {
|
||||||
let ptr = unsafe { libdeno::deno_last_exception(self.libdeno_isolate) };
|
let ptr = unsafe { libdeno::deno_last_exception(self.libdeno_isolate) };
|
||||||
if ptr.is_null() {
|
if ptr.is_null() {
|
||||||
|
@ -618,7 +619,8 @@ mod tests {
|
||||||
fn test_dispatch_sync() {
|
fn test_dispatch_sync() {
|
||||||
let state = IsolateState::mock();
|
let state = IsolateState::mock();
|
||||||
let snapshot = libdeno::deno_buf::empty();
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
let isolate = Isolate::new(snapshot, state, dispatch_sync);
|
let permissions = DenoPermissions::default();
|
||||||
|
let isolate = Isolate::new(snapshot, state, dispatch_sync, permissions);
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
isolate
|
isolate
|
||||||
.execute(
|
.execute(
|
||||||
|
@ -657,7 +659,9 @@ mod tests {
|
||||||
fn test_metrics_sync() {
|
fn test_metrics_sync() {
|
||||||
let state = IsolateState::mock();
|
let state = IsolateState::mock();
|
||||||
let snapshot = libdeno::deno_buf::empty();
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
let isolate = Isolate::new(snapshot, state, metrics_dispatch_sync);
|
let permissions = DenoPermissions::default();
|
||||||
|
let isolate =
|
||||||
|
Isolate::new(snapshot, state, metrics_dispatch_sync, permissions);
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
// Verify that metrics have been properly initialized.
|
// Verify that metrics have been properly initialized.
|
||||||
{
|
{
|
||||||
|
@ -691,7 +695,9 @@ mod tests {
|
||||||
fn test_metrics_async() {
|
fn test_metrics_async() {
|
||||||
let state = IsolateState::mock();
|
let state = IsolateState::mock();
|
||||||
let snapshot = libdeno::deno_buf::empty();
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
let isolate = Isolate::new(snapshot, state, metrics_dispatch_async);
|
let permissions = DenoPermissions::default();
|
||||||
|
let isolate =
|
||||||
|
Isolate::new(snapshot, state, metrics_dispatch_async, permissions);
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
// Verify that metrics have been properly initialized.
|
// Verify that metrics have been properly initialized.
|
||||||
{
|
{
|
||||||
|
@ -779,7 +785,8 @@ mod tests {
|
||||||
|
|
||||||
let state = Arc::new(IsolateState::new(flags, rest_argv, None));
|
let state = Arc::new(IsolateState::new(flags, rest_argv, None));
|
||||||
let snapshot = libdeno::deno_buf::empty();
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
let mut isolate = Isolate::new(snapshot, state, dispatch_sync);
|
let permissions = DenoPermissions::default();
|
||||||
|
let mut isolate = Isolate::new(snapshot, state, dispatch_sync, permissions);
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
isolate
|
isolate
|
||||||
.execute_mod(filename, false)
|
.execute_mod(filename, false)
|
||||||
|
@ -801,7 +808,8 @@ mod tests {
|
||||||
|
|
||||||
let state = Arc::new(IsolateState::new(flags, rest_argv, None));
|
let state = Arc::new(IsolateState::new(flags, rest_argv, None));
|
||||||
let snapshot = libdeno::deno_buf::empty();
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
let mut isolate = Isolate::new(snapshot, state, dispatch_sync);
|
let permissions = DenoPermissions::default();
|
||||||
|
let mut isolate = Isolate::new(snapshot, state, dispatch_sync, permissions);
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
isolate
|
isolate
|
||||||
.execute_mod(filename, false)
|
.execute_mod(filename, false)
|
||||||
|
|
|
@ -96,7 +96,9 @@ fn main() {
|
||||||
|
|
||||||
let state = Arc::new(isolate::IsolateState::new(flags, rest_argv, None));
|
let state = Arc::new(isolate::IsolateState::new(flags, rest_argv, None));
|
||||||
let snapshot = snapshot::deno_snapshot();
|
let snapshot = snapshot::deno_snapshot();
|
||||||
let mut isolate = isolate::Isolate::new(snapshot, state, ops::dispatch);
|
let permissions = permissions::DenoPermissions::from_flags(&state.flags);
|
||||||
|
let mut isolate =
|
||||||
|
isolate::Isolate::new(snapshot, state, ops::dispatch, permissions);
|
||||||
|
|
||||||
tokio_util::init(|| {
|
tokio_util::init(|| {
|
||||||
// Setup runtime.
|
// Setup runtime.
|
||||||
|
|
314
src/ops.rs
314
src/ops.rs
|
@ -3,7 +3,7 @@
|
||||||
use atty;
|
use atty;
|
||||||
use crate::ansi;
|
use crate::ansi;
|
||||||
use crate::errors;
|
use crate::errors;
|
||||||
use crate::errors::{DenoError, DenoResult, ErrorKind};
|
use crate::errors::{permission_denied, DenoError, DenoResult, ErrorKind};
|
||||||
use crate::fs as deno_fs;
|
use crate::fs as deno_fs;
|
||||||
use crate::http_util;
|
use crate::http_util;
|
||||||
use crate::isolate::Buf;
|
use crate::isolate::Buf;
|
||||||
|
@ -39,6 +39,7 @@ use std::net::Shutdown;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::process::Command;
|
use std::process::Command;
|
||||||
|
use std::sync::atomic::Ordering;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::{Duration, Instant, SystemTime, UNIX_EPOCH};
|
use std::time::{Duration, Instant, SystemTime, UNIX_EPOCH};
|
||||||
use tokio;
|
use tokio;
|
||||||
|
@ -57,7 +58,7 @@ type OpResult = DenoResult<Buf>;
|
||||||
// TODO Ideally we wouldn't have to box the Op being returned.
|
// TODO Ideally we wouldn't have to box the Op being returned.
|
||||||
// The box is just to make it easier to get a prototype refactor working.
|
// The box is just to make it easier to get a prototype refactor working.
|
||||||
type OpCreator =
|
type OpCreator =
|
||||||
fn(state: &Arc<IsolateState>, base: &msg::Base<'_>, data: libdeno::deno_buf)
|
fn(isolate: &Isolate, base: &msg::Base<'_>, data: libdeno::deno_buf)
|
||||||
-> Box<Op>;
|
-> Box<Op>;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
@ -134,7 +135,7 @@ pub fn dispatch(
|
||||||
msg::enum_name_any(inner_type)
|
msg::enum_name_any(inner_type)
|
||||||
)),
|
)),
|
||||||
};
|
};
|
||||||
op_creator(&isolate.state, &base, data)
|
op_creator(&isolate, &base, data)
|
||||||
};
|
};
|
||||||
|
|
||||||
let boxed_op = Box::new(
|
let boxed_op = Box::new(
|
||||||
|
@ -182,7 +183,7 @@ pub fn dispatch(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_now(
|
fn op_now(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -206,7 +207,7 @@ fn op_now(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_is_tty(
|
fn op_is_tty(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
_data: libdeno::deno_buf,
|
_data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -231,7 +232,7 @@ fn op_is_tty(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_exit(
|
fn op_exit(
|
||||||
_config: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
_data: libdeno::deno_buf,
|
_data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -240,14 +241,19 @@ fn op_exit(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_start(
|
fn op_start(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert_eq!(data.len(), 0);
|
assert_eq!(data.len(), 0);
|
||||||
let mut builder = FlatBufferBuilder::new();
|
let mut builder = FlatBufferBuilder::new();
|
||||||
|
|
||||||
let argv = state.argv.iter().map(|s| s.as_str()).collect::<Vec<_>>();
|
let argv = isolate
|
||||||
|
.state
|
||||||
|
.argv
|
||||||
|
.iter()
|
||||||
|
.map(|s| s.as_str())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
let argv_off = builder.create_vector_of_strings(argv.as_slice());
|
let argv_off = builder.create_vector_of_strings(argv.as_slice());
|
||||||
|
|
||||||
let cwd_path = std::env::current_dir().unwrap();
|
let cwd_path = std::env::current_dir().unwrap();
|
||||||
|
@ -263,7 +269,10 @@ fn op_start(
|
||||||
let deno_version = version::DENO;
|
let deno_version = version::DENO;
|
||||||
let deno_version_off = builder.create_string(deno_version);
|
let deno_version_off = builder.create_string(deno_version);
|
||||||
|
|
||||||
let main_module = state.main_module().map(|m| builder.create_string(&m));
|
let main_module = isolate
|
||||||
|
.state
|
||||||
|
.main_module()
|
||||||
|
.map(|m| builder.create_string(&m));
|
||||||
|
|
||||||
let inner = msg::StartRes::create(
|
let inner = msg::StartRes::create(
|
||||||
&mut builder,
|
&mut builder,
|
||||||
|
@ -272,9 +281,9 @@ fn op_start(
|
||||||
pid: std::process::id(),
|
pid: std::process::id(),
|
||||||
argv: Some(argv_off),
|
argv: Some(argv_off),
|
||||||
main_module,
|
main_module,
|
||||||
debug_flag: state.flags.log_debug,
|
debug_flag: isolate.state.flags.log_debug,
|
||||||
types_flag: state.flags.types,
|
types_flag: isolate.state.flags.types,
|
||||||
version_flag: state.flags.version,
|
version_flag: isolate.state.flags.version,
|
||||||
v8_version: Some(v8_version_off),
|
v8_version: Some(v8_version_off),
|
||||||
deno_version: Some(deno_version_off),
|
deno_version: Some(deno_version_off),
|
||||||
no_color: !ansi::use_color(),
|
no_color: !ansi::use_color(),
|
||||||
|
@ -295,7 +304,7 @@ fn op_start(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_format_error(
|
fn op_format_error(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -304,7 +313,7 @@ fn op_format_error(
|
||||||
let orig_error = String::from(inner.error().unwrap());
|
let orig_error = String::from(inner.error().unwrap());
|
||||||
|
|
||||||
let js_error = JSError::from_v8_exception(&orig_error).unwrap();
|
let js_error = JSError::from_v8_exception(&orig_error).unwrap();
|
||||||
let js_error_mapped = apply_source_map(&js_error, &state.dir);
|
let js_error_mapped = apply_source_map(&js_error, &isolate.state.dir);
|
||||||
let js_error_string = JSErrorColor(&js_error_mapped).to_string();
|
let js_error_string = JSErrorColor(&js_error_mapped).to_string();
|
||||||
|
|
||||||
let mut builder = FlatBufferBuilder::new();
|
let mut builder = FlatBufferBuilder::new();
|
||||||
|
@ -355,7 +364,7 @@ pub fn odd_future(err: DenoError) -> Box<Op> {
|
||||||
|
|
||||||
// https://github.com/denoland/deno/blob/golang/os.go#L100-L154
|
// https://github.com/denoland/deno/blob/golang/os.go#L100-L154
|
||||||
fn op_fetch_module_meta_data(
|
fn op_fetch_module_meta_data(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -365,11 +374,28 @@ fn op_fetch_module_meta_data(
|
||||||
let specifier = inner.specifier().unwrap();
|
let specifier = inner.specifier().unwrap();
|
||||||
let referrer = inner.referrer().unwrap();
|
let referrer = inner.referrer().unwrap();
|
||||||
|
|
||||||
assert_eq!(state.dir.root.join("gen"), state.dir.gen, "Sanity check");
|
if !isolate.permissions.allow_read.load(Ordering::SeqCst) {
|
||||||
|
debug!("No read permission for fetch_module_meta_data");
|
||||||
|
return odd_future(permission_denied());
|
||||||
|
}
|
||||||
|
|
||||||
|
if !isolate.permissions.allow_net.load(Ordering::SeqCst) {
|
||||||
|
debug!("No network permission for fetch_module_meta_data");
|
||||||
|
return odd_future(permission_denied());
|
||||||
|
}
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
isolate.state.dir.root.join("gen"),
|
||||||
|
isolate.state.dir.gen,
|
||||||
|
"Sanity check"
|
||||||
|
);
|
||||||
|
|
||||||
Box::new(futures::future::result(|| -> OpResult {
|
Box::new(futures::future::result(|| -> OpResult {
|
||||||
let builder = &mut FlatBufferBuilder::new();
|
let builder = &mut FlatBufferBuilder::new();
|
||||||
let out = state.dir.fetch_module_meta_data(specifier, referrer)?;
|
let out = isolate
|
||||||
|
.state
|
||||||
|
.dir
|
||||||
|
.fetch_module_meta_data(specifier, referrer)?;
|
||||||
let data_off = builder.create_vector(out.source_code.as_slice());
|
let data_off = builder.create_vector(out.source_code.as_slice());
|
||||||
let msg_args = msg::FetchModuleMetaDataResArgs {
|
let msg_args = msg::FetchModuleMetaDataResArgs {
|
||||||
module_name: Some(builder.create_string(&out.module_name)),
|
module_name: Some(builder.create_string(&out.module_name)),
|
||||||
|
@ -391,7 +417,7 @@ fn op_fetch_module_meta_data(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_chdir(
|
fn op_chdir(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -422,7 +448,7 @@ fn op_set_timeout(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_set_env(
|
fn op_set_env(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -430,7 +456,7 @@ fn op_set_env(
|
||||||
let inner = base.inner_as_set_env().unwrap();
|
let inner = base.inner_as_set_env().unwrap();
|
||||||
let key = inner.key().unwrap();
|
let key = inner.key().unwrap();
|
||||||
let value = inner.value().unwrap();
|
let value = inner.value().unwrap();
|
||||||
if let Err(e) = state.check_env() {
|
if let Err(e) = isolate.check_env() {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
std::env::set_var(key, value);
|
std::env::set_var(key, value);
|
||||||
|
@ -438,14 +464,14 @@ fn op_set_env(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_env(
|
fn op_env(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert_eq!(data.len(), 0);
|
assert_eq!(data.len(), 0);
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
|
||||||
if let Err(e) = state.check_env() {
|
if let Err(e) = isolate.check_env() {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -470,7 +496,7 @@ fn op_env(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_fetch(
|
fn op_fetch(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -493,7 +519,7 @@ fn op_fetch(
|
||||||
}
|
}
|
||||||
let req = maybe_req.unwrap();
|
let req = maybe_req.unwrap();
|
||||||
|
|
||||||
if let Err(e) = state.check_net(url) {
|
if let Err(e) = isolate.check_net(url) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -557,7 +583,7 @@ where
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_make_temp_dir(
|
fn op_make_temp_dir(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -567,7 +593,7 @@ fn op_make_temp_dir(
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
|
||||||
// FIXME
|
// FIXME
|
||||||
if let Err(e) = state.check_write("make_temp") {
|
if let Err(e) = isolate.check_write("make_temp") {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -606,7 +632,7 @@ fn op_make_temp_dir(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_mkdir(
|
fn op_mkdir(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -616,7 +642,7 @@ fn op_mkdir(
|
||||||
let recursive = inner.recursive();
|
let recursive = inner.recursive();
|
||||||
let mode = inner.mode();
|
let mode = inner.mode();
|
||||||
|
|
||||||
if let Err(e) = state.check_write(&path) {
|
if let Err(e) = isolate.check_write(&path) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -628,7 +654,7 @@ fn op_mkdir(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_chmod(
|
fn op_chmod(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -637,7 +663,7 @@ fn op_chmod(
|
||||||
let _mode = inner.mode();
|
let _mode = inner.mode();
|
||||||
let path = String::from(inner.path().unwrap());
|
let path = String::from(inner.path().unwrap());
|
||||||
|
|
||||||
if let Err(e) = state.check_write(&path) {
|
if let Err(e) = isolate.check_write(&path) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -667,7 +693,7 @@ fn op_chmod(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_open(
|
fn op_open(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -716,20 +742,20 @@ fn op_open(
|
||||||
|
|
||||||
match mode {
|
match mode {
|
||||||
"r" => {
|
"r" => {
|
||||||
if let Err(e) = state.check_read(&filename_str) {
|
if let Err(e) = isolate.check_read(&filename_str) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
"w" | "a" | "x" => {
|
"w" | "a" | "x" => {
|
||||||
if let Err(e) = state.check_write(&filename_str) {
|
if let Err(e) = isolate.check_write(&filename_str) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
&_ => {
|
&_ => {
|
||||||
if let Err(e) = state.check_read(&filename_str) {
|
if let Err(e) = isolate.check_read(&filename_str) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
if let Err(e) = state.check_write(&filename_str) {
|
if let Err(e) = isolate.check_write(&filename_str) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -757,7 +783,7 @@ fn op_open(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_close(
|
fn op_close(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -774,7 +800,7 @@ fn op_close(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_shutdown(
|
fn op_shutdown(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -800,7 +826,7 @@ fn op_shutdown(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_read(
|
fn op_read(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -838,7 +864,7 @@ fn op_read(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_write(
|
fn op_write(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -875,7 +901,7 @@ fn op_write(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_seek(
|
fn op_seek(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -897,7 +923,7 @@ fn op_seek(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_remove(
|
fn op_remove(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -907,7 +933,7 @@ fn op_remove(
|
||||||
let path = PathBuf::from(path_);
|
let path = PathBuf::from(path_);
|
||||||
let recursive = inner.recursive();
|
let recursive = inner.recursive();
|
||||||
|
|
||||||
if let Err(e) = state.check_write(path.to_str().unwrap()) {
|
if let Err(e) = isolate.check_write(path.to_str().unwrap()) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -927,7 +953,7 @@ fn op_remove(
|
||||||
|
|
||||||
// Prototype https://github.com/denoland/deno/blob/golang/os.go#L171-L184
|
// Prototype https://github.com/denoland/deno/blob/golang/os.go#L171-L184
|
||||||
fn op_read_file(
|
fn op_read_file(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -937,7 +963,7 @@ fn op_read_file(
|
||||||
let filename_ = inner.filename().unwrap();
|
let filename_ = inner.filename().unwrap();
|
||||||
let filename = PathBuf::from(filename_);
|
let filename = PathBuf::from(filename_);
|
||||||
debug!("op_read_file {}", filename.display());
|
debug!("op_read_file {}", filename.display());
|
||||||
if let Err(e) = state.check_read(&filename_) {
|
if let Err(e) = isolate.check_read(&filename_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
blocking(base.sync(), move || {
|
blocking(base.sync(), move || {
|
||||||
|
@ -965,7 +991,7 @@ fn op_read_file(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_copy_file(
|
fn op_copy_file(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -976,10 +1002,10 @@ fn op_copy_file(
|
||||||
let to_ = inner.to().unwrap();
|
let to_ = inner.to().unwrap();
|
||||||
let to = PathBuf::from(to_);
|
let to = PathBuf::from(to_);
|
||||||
|
|
||||||
if let Err(e) = state.check_read(&from_) {
|
if let Err(e) = isolate.check_read(&from_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
if let Err(e) = state.check_write(&to_) {
|
if let Err(e) = isolate.check_write(&to_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1021,7 +1047,7 @@ fn get_mode(_perm: &fs::Permissions) -> u32 {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_cwd(
|
fn op_cwd(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1047,7 +1073,7 @@ fn op_cwd(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_stat(
|
fn op_stat(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1058,7 +1084,7 @@ fn op_stat(
|
||||||
let filename = PathBuf::from(filename_);
|
let filename = PathBuf::from(filename_);
|
||||||
let lstat = inner.lstat();
|
let lstat = inner.lstat();
|
||||||
|
|
||||||
if let Err(e) = state.check_read(&filename_) {
|
if let Err(e) = isolate.check_read(&filename_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1099,7 +1125,7 @@ fn op_stat(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_read_dir(
|
fn op_read_dir(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1108,7 +1134,7 @@ fn op_read_dir(
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
let path = String::from(inner.path().unwrap());
|
let path = String::from(inner.path().unwrap());
|
||||||
|
|
||||||
if let Err(e) = state.check_read(&path) {
|
if let Err(e) = isolate.check_read(&path) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1160,7 +1186,7 @@ fn op_read_dir(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_write_file(
|
fn op_write_file(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1171,7 +1197,7 @@ fn op_write_file(
|
||||||
let is_create = inner.is_create();
|
let is_create = inner.is_create();
|
||||||
let is_append = inner.is_append();
|
let is_append = inner.is_append();
|
||||||
|
|
||||||
if let Err(e) = state.check_write(&filename) {
|
if let Err(e) = isolate.check_write(&filename) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1190,7 +1216,7 @@ fn op_write_file(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_rename(
|
fn op_rename(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1199,7 +1225,7 @@ fn op_rename(
|
||||||
let oldpath = PathBuf::from(inner.oldpath().unwrap());
|
let oldpath = PathBuf::from(inner.oldpath().unwrap());
|
||||||
let newpath_ = inner.newpath().unwrap();
|
let newpath_ = inner.newpath().unwrap();
|
||||||
let newpath = PathBuf::from(newpath_);
|
let newpath = PathBuf::from(newpath_);
|
||||||
if let Err(e) = state.check_write(&newpath_) {
|
if let Err(e) = isolate.check_write(&newpath_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
blocking(base.sync(), move || -> OpResult {
|
blocking(base.sync(), move || -> OpResult {
|
||||||
|
@ -1210,7 +1236,7 @@ fn op_rename(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_symlink(
|
fn op_symlink(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1220,7 +1246,7 @@ fn op_symlink(
|
||||||
let newname_ = inner.newname().unwrap();
|
let newname_ = inner.newname().unwrap();
|
||||||
let newname = PathBuf::from(newname_);
|
let newname = PathBuf::from(newname_);
|
||||||
|
|
||||||
if let Err(e) = state.check_write(&newname_) {
|
if let Err(e) = isolate.check_write(&newname_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
// TODO Use type for Windows.
|
// TODO Use type for Windows.
|
||||||
|
@ -1239,7 +1265,7 @@ fn op_symlink(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_read_link(
|
fn op_read_link(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1249,7 +1275,7 @@ fn op_read_link(
|
||||||
let name_ = inner.name().unwrap();
|
let name_ = inner.name().unwrap();
|
||||||
let name = PathBuf::from(name_);
|
let name = PathBuf::from(name_);
|
||||||
|
|
||||||
if let Err(e) = state.check_read(&name_) {
|
if let Err(e) = isolate.check_read(&name_) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1277,7 +1303,7 @@ fn op_read_link(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_repl_start(
|
fn op_repl_start(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1287,7 +1313,7 @@ fn op_repl_start(
|
||||||
let history_file = String::from(inner.history_file().unwrap());
|
let history_file = String::from(inner.history_file().unwrap());
|
||||||
|
|
||||||
debug!("op_repl_start {}", history_file);
|
debug!("op_repl_start {}", history_file);
|
||||||
let history_path = repl::history_path(&state.dir, &history_file);
|
let history_path = repl::history_path(&isolate.state.dir, &history_file);
|
||||||
let repl = repl::Repl::new(history_path);
|
let repl = repl::Repl::new(history_path);
|
||||||
let resource = resources::add_repl(repl);
|
let resource = resources::add_repl(repl);
|
||||||
|
|
||||||
|
@ -1308,7 +1334,7 @@ fn op_repl_start(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_repl_readline(
|
fn op_repl_readline(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1344,7 +1370,7 @@ fn op_repl_readline(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_truncate(
|
fn op_truncate(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1354,7 +1380,7 @@ fn op_truncate(
|
||||||
let filename = String::from(inner.name().unwrap());
|
let filename = String::from(inner.name().unwrap());
|
||||||
let len = inner.len();
|
let len = inner.len();
|
||||||
|
|
||||||
if let Err(e) = state.check_write(&filename) {
|
if let Err(e) = isolate.check_write(&filename) {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1367,12 +1393,12 @@ fn op_truncate(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_listen(
|
fn op_listen(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert_eq!(data.len(), 0);
|
assert_eq!(data.len(), 0);
|
||||||
if let Err(e) = state.check_net("listen") {
|
if let Err(e) = isolate.check_net("listen") {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1429,12 +1455,12 @@ fn new_conn(cmd_id: u32, tcp_stream: TcpStream) -> OpResult {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_accept(
|
fn op_accept(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert_eq!(data.len(), 0);
|
assert_eq!(data.len(), 0);
|
||||||
if let Err(e) = state.check_net("accept") {
|
if let Err(e) = isolate.check_net("accept") {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
@ -1455,12 +1481,12 @@ fn op_accept(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_dial(
|
fn op_dial(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert_eq!(data.len(), 0);
|
assert_eq!(data.len(), 0);
|
||||||
if let Err(e) = state.check_net("dial") {
|
if let Err(e) = isolate.check_net("dial") {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
@ -1481,7 +1507,7 @@ fn op_dial(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_metrics(
|
fn op_metrics(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1491,7 +1517,7 @@ fn op_metrics(
|
||||||
let builder = &mut FlatBufferBuilder::new();
|
let builder = &mut FlatBufferBuilder::new();
|
||||||
let inner = msg::MetricsRes::create(
|
let inner = msg::MetricsRes::create(
|
||||||
builder,
|
builder,
|
||||||
&msg::MetricsResArgs::from(&state.metrics),
|
&msg::MetricsResArgs::from(&isolate.state.metrics),
|
||||||
);
|
);
|
||||||
ok_future(serialize_response(
|
ok_future(serialize_response(
|
||||||
cmd_id,
|
cmd_id,
|
||||||
|
@ -1505,7 +1531,7 @@ fn op_metrics(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_resources(
|
fn op_resources(
|
||||||
_state: &Arc<IsolateState>,
|
_isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1557,14 +1583,14 @@ fn subprocess_stdio_map(v: msg::ProcessStdio) -> std::process::Stdio {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_run(
|
fn op_run(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
assert!(base.sync());
|
assert!(base.sync());
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
|
||||||
if let Err(e) = state.check_run() {
|
if let Err(e) = isolate.check_run() {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1630,7 +1656,7 @@ fn op_run(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_run_status(
|
fn op_run_status(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1639,7 +1665,7 @@ fn op_run_status(
|
||||||
let inner = base.inner_as_run_status().unwrap();
|
let inner = base.inner_as_run_status().unwrap();
|
||||||
let rid = inner.rid();
|
let rid = inner.rid();
|
||||||
|
|
||||||
if let Err(e) = state.check_run() {
|
if let Err(e) = isolate.check_run() {
|
||||||
return odd_future(e);
|
return odd_future(e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1706,7 +1732,7 @@ impl Future for GetMessageFuture {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_worker_get_message(
|
fn op_worker_get_message(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1714,7 +1740,7 @@ fn op_worker_get_message(
|
||||||
let cmd_id = base.cmd_id();
|
let cmd_id = base.cmd_id();
|
||||||
|
|
||||||
let op = GetMessageFuture {
|
let op = GetMessageFuture {
|
||||||
state: state.clone(),
|
state: isolate.state.clone(),
|
||||||
};
|
};
|
||||||
let op = op.map_err(move |_| -> DenoError { unimplemented!() });
|
let op = op.map_err(move |_| -> DenoError { unimplemented!() });
|
||||||
let op = op.and_then(move |maybe_buf| -> DenoResult<Buf> {
|
let op = op.and_then(move |maybe_buf| -> DenoResult<Buf> {
|
||||||
|
@ -1740,7 +1766,7 @@ fn op_worker_get_message(
|
||||||
}
|
}
|
||||||
|
|
||||||
fn op_worker_post_message(
|
fn op_worker_post_message(
|
||||||
state: &Arc<IsolateState>,
|
isolate: &Isolate,
|
||||||
base: &msg::Base<'_>,
|
base: &msg::Base<'_>,
|
||||||
data: libdeno::deno_buf,
|
data: libdeno::deno_buf,
|
||||||
) -> Box<Op> {
|
) -> Box<Op> {
|
||||||
|
@ -1748,8 +1774,8 @@ fn op_worker_post_message(
|
||||||
|
|
||||||
let d = Vec::from(data.as_ref()).into_boxed_slice();
|
let d = Vec::from(data.as_ref()).into_boxed_slice();
|
||||||
|
|
||||||
assert!(state.worker_channels.is_some());
|
assert!(isolate.state.worker_channels.is_some());
|
||||||
let tx = match state.worker_channels {
|
let tx = match isolate.state.worker_channels {
|
||||||
None => panic!("expected worker_channels"),
|
None => panic!("expected worker_channels"),
|
||||||
Some(ref wc) => {
|
Some(ref wc) => {
|
||||||
let wc = wc.lock().unwrap();
|
let wc = wc.lock().unwrap();
|
||||||
|
@ -1771,3 +1797,125 @@ fn op_worker_post_message(
|
||||||
});
|
});
|
||||||
Box::new(op)
|
Box::new(op)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
use crate::isolate::{Isolate, IsolateState};
|
||||||
|
use crate::permissions::DenoPermissions;
|
||||||
|
use std::sync::atomic::AtomicBool;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn fetch_module_meta_fails_without_read() {
|
||||||
|
let state = IsolateState::mock();
|
||||||
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
|
let permissions = DenoPermissions {
|
||||||
|
allow_read: AtomicBool::new(false),
|
||||||
|
allow_write: AtomicBool::new(true),
|
||||||
|
allow_env: AtomicBool::new(true),
|
||||||
|
allow_net: AtomicBool::new(true),
|
||||||
|
allow_run: AtomicBool::new(true),
|
||||||
|
};
|
||||||
|
let isolate = Isolate::new(snapshot, state, dispatch, permissions);
|
||||||
|
let builder = &mut FlatBufferBuilder::new();
|
||||||
|
let fetch_msg_args = msg::FetchModuleMetaDataArgs {
|
||||||
|
specifier: Some(builder.create_string("./somefile")),
|
||||||
|
referrer: Some(builder.create_string(".")),
|
||||||
|
};
|
||||||
|
let inner = msg::FetchModuleMetaData::create(builder, &fetch_msg_args);
|
||||||
|
let base_args = msg::BaseArgs {
|
||||||
|
inner: Some(inner.as_union_value()),
|
||||||
|
inner_type: msg::Any::FetchModuleMetaData,
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
let base = msg::Base::create(builder, &base_args);
|
||||||
|
msg::finish_base_buffer(builder, base);
|
||||||
|
let data = builder.finished_data();
|
||||||
|
let final_msg = msg::get_root_as_base(&data);
|
||||||
|
let fetch_result = op_fetch_module_meta_data(
|
||||||
|
&isolate,
|
||||||
|
&final_msg,
|
||||||
|
libdeno::deno_buf::empty(),
|
||||||
|
).wait();
|
||||||
|
match fetch_result {
|
||||||
|
Ok(_) => assert!(true),
|
||||||
|
Err(e) => assert_eq!(e.to_string(), permission_denied().to_string()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn fetch_module_meta_fails_without_net() {
|
||||||
|
let state = IsolateState::mock();
|
||||||
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
|
let permissions = DenoPermissions {
|
||||||
|
allow_read: AtomicBool::new(true),
|
||||||
|
allow_write: AtomicBool::new(true),
|
||||||
|
allow_env: AtomicBool::new(true),
|
||||||
|
allow_net: AtomicBool::new(false),
|
||||||
|
allow_run: AtomicBool::new(true),
|
||||||
|
};
|
||||||
|
let isolate = Isolate::new(snapshot, state, dispatch, permissions);
|
||||||
|
let builder = &mut FlatBufferBuilder::new();
|
||||||
|
let fetch_msg_args = msg::FetchModuleMetaDataArgs {
|
||||||
|
specifier: Some(builder.create_string("./somefile")),
|
||||||
|
referrer: Some(builder.create_string(".")),
|
||||||
|
};
|
||||||
|
let inner = msg::FetchModuleMetaData::create(builder, &fetch_msg_args);
|
||||||
|
let base_args = msg::BaseArgs {
|
||||||
|
inner: Some(inner.as_union_value()),
|
||||||
|
inner_type: msg::Any::FetchModuleMetaData,
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
let base = msg::Base::create(builder, &base_args);
|
||||||
|
msg::finish_base_buffer(builder, base);
|
||||||
|
let data = builder.finished_data();
|
||||||
|
let final_msg = msg::get_root_as_base(&data);
|
||||||
|
let fetch_result = op_fetch_module_meta_data(
|
||||||
|
&isolate,
|
||||||
|
&final_msg,
|
||||||
|
libdeno::deno_buf::empty(),
|
||||||
|
).wait();
|
||||||
|
match fetch_result {
|
||||||
|
Ok(_) => assert!(true),
|
||||||
|
Err(e) => assert_eq!(e.to_string(), permission_denied().to_string()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn fetch_module_meta_not_permission_denied_with_permissions() {
|
||||||
|
let state = IsolateState::mock();
|
||||||
|
let snapshot = libdeno::deno_buf::empty();
|
||||||
|
let permissions = DenoPermissions {
|
||||||
|
allow_read: AtomicBool::new(true),
|
||||||
|
allow_write: AtomicBool::new(false),
|
||||||
|
allow_env: AtomicBool::new(false),
|
||||||
|
allow_net: AtomicBool::new(true),
|
||||||
|
allow_run: AtomicBool::new(false),
|
||||||
|
};
|
||||||
|
let isolate = Isolate::new(snapshot, state, dispatch, permissions);
|
||||||
|
let builder = &mut FlatBufferBuilder::new();
|
||||||
|
let fetch_msg_args = msg::FetchModuleMetaDataArgs {
|
||||||
|
specifier: Some(builder.create_string("./somefile")),
|
||||||
|
referrer: Some(builder.create_string(".")),
|
||||||
|
};
|
||||||
|
let inner = msg::FetchModuleMetaData::create(builder, &fetch_msg_args);
|
||||||
|
let base_args = msg::BaseArgs {
|
||||||
|
inner: Some(inner.as_union_value()),
|
||||||
|
inner_type: msg::Any::FetchModuleMetaData,
|
||||||
|
..Default::default()
|
||||||
|
};
|
||||||
|
let base = msg::Base::create(builder, &base_args);
|
||||||
|
msg::finish_base_buffer(builder, base);
|
||||||
|
let data = builder.finished_data();
|
||||||
|
let final_msg = msg::get_root_as_base(&data);
|
||||||
|
let fetch_result = op_fetch_module_meta_data(
|
||||||
|
&isolate,
|
||||||
|
&final_msg,
|
||||||
|
libdeno::deno_buf::empty(),
|
||||||
|
).wait();
|
||||||
|
match fetch_result {
|
||||||
|
Ok(_) => assert!(true),
|
||||||
|
Err(e) => assert!(e.to_string() != permission_denied().to_string()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -20,7 +20,7 @@ pub struct DenoPermissions {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DenoPermissions {
|
impl DenoPermissions {
|
||||||
pub fn new(flags: &DenoFlags) -> Self {
|
pub fn from_flags(flags: &DenoFlags) -> Self {
|
||||||
Self {
|
Self {
|
||||||
allow_read: AtomicBool::new(flags.allow_read),
|
allow_read: AtomicBool::new(flags.allow_read),
|
||||||
allow_write: AtomicBool::new(flags.allow_write),
|
allow_write: AtomicBool::new(flags.allow_write),
|
||||||
|
@ -90,6 +90,16 @@ impl DenoPermissions {
|
||||||
}
|
}
|
||||||
r
|
r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
allow_read: AtomicBool::new(false),
|
||||||
|
allow_write: AtomicBool::new(false),
|
||||||
|
allow_env: AtomicBool::new(false),
|
||||||
|
allow_net: AtomicBool::new(false),
|
||||||
|
allow_run: AtomicBool::new(false),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn permission_prompt(message: &str) -> DenoResult<()> {
|
fn permission_prompt(message: &str) -> DenoResult<()> {
|
||||||
|
|
|
@ -5,6 +5,7 @@ use crate::isolate::IsolateState;
|
||||||
use crate::isolate::WorkerChannels;
|
use crate::isolate::WorkerChannels;
|
||||||
use crate::js_errors::JSErrorColor;
|
use crate::js_errors::JSErrorColor;
|
||||||
use crate::ops;
|
use crate::ops;
|
||||||
|
use crate::permissions::DenoPermissions;
|
||||||
use crate::resources;
|
use crate::resources;
|
||||||
use crate::snapshot;
|
use crate::snapshot;
|
||||||
use crate::tokio_util;
|
use crate::tokio_util;
|
||||||
|
@ -22,7 +23,10 @@ pub struct Worker {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Worker {
|
impl Worker {
|
||||||
pub fn new(parent_state: &Arc<IsolateState>) -> (Self, WorkerChannels) {
|
pub fn new(
|
||||||
|
parent_state: &Arc<IsolateState>,
|
||||||
|
permissions: DenoPermissions,
|
||||||
|
) -> (Self, WorkerChannels) {
|
||||||
let (worker_in_tx, worker_in_rx) = mpsc::channel::<Buf>(1);
|
let (worker_in_tx, worker_in_rx) = mpsc::channel::<Buf>(1);
|
||||||
let (worker_out_tx, worker_out_rx) = mpsc::channel::<Buf>(1);
|
let (worker_out_tx, worker_out_rx) = mpsc::channel::<Buf>(1);
|
||||||
|
|
||||||
|
@ -36,7 +40,7 @@ impl Worker {
|
||||||
));
|
));
|
||||||
|
|
||||||
let snapshot = snapshot::compiler_snapshot();
|
let snapshot = snapshot::compiler_snapshot();
|
||||||
let isolate = Isolate::new(snapshot, state, ops::dispatch);
|
let isolate = Isolate::new(snapshot, state, ops::dispatch, permissions);
|
||||||
|
|
||||||
let worker = Worker { isolate };
|
let worker = Worker { isolate };
|
||||||
(worker, external_channels)
|
(worker, external_channels)
|
||||||
|
@ -54,6 +58,7 @@ impl Worker {
|
||||||
pub fn spawn(
|
pub fn spawn(
|
||||||
state: Arc<IsolateState>,
|
state: Arc<IsolateState>,
|
||||||
js_source: String,
|
js_source: String,
|
||||||
|
permissions: DenoPermissions,
|
||||||
) -> resources::Resource {
|
) -> resources::Resource {
|
||||||
// TODO This function should return a Future, so that the caller can retrieve
|
// TODO This function should return a Future, so that the caller can retrieve
|
||||||
// the JSError if one is thrown. Currently it just prints to stderr and calls
|
// the JSError if one is thrown. Currently it just prints to stderr and calls
|
||||||
|
@ -63,7 +68,7 @@ pub fn spawn(
|
||||||
let builder = thread::Builder::new().name("worker".to_string());
|
let builder = thread::Builder::new().name("worker".to_string());
|
||||||
let _tid = builder
|
let _tid = builder
|
||||||
.spawn(move || {
|
.spawn(move || {
|
||||||
let (worker, external_channels) = Worker::new(&state);
|
let (worker, external_channels) = Worker::new(&state, permissions);
|
||||||
|
|
||||||
let resource = resources::add_worker(external_channels);
|
let resource = resources::add_worker(external_channels);
|
||||||
p.send(resource.clone()).unwrap();
|
p.send(resource.clone()).unwrap();
|
||||||
|
@ -109,6 +114,7 @@ mod tests {
|
||||||
console.log("after postMessage");
|
console.log("after postMessage");
|
||||||
}
|
}
|
||||||
"#.into(),
|
"#.into(),
|
||||||
|
DenoPermissions::default(),
|
||||||
);
|
);
|
||||||
let msg = String::from("hi").into_boxed_str().into_boxed_bytes();
|
let msg = String::from("hi").into_boxed_str().into_boxed_bytes();
|
||||||
|
|
||||||
|
@ -127,8 +133,11 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn removed_from_resource_table_on_close() {
|
fn removed_from_resource_table_on_close() {
|
||||||
let resource =
|
let resource = spawn(
|
||||||
spawn(IsolateState::mock(), "onmessage = () => close();".into());
|
IsolateState::mock(),
|
||||||
|
"onmessage = () => close();".into(),
|
||||||
|
DenoPermissions::default(),
|
||||||
|
);
|
||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
resources::get_type(resource.rid),
|
resources::get_type(resource.rid),
|
||||||
|
|
Loading…
Reference in a new issue