// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license. use crate::file_fetcher::SourceFileFetcher; use crate::global_state::GlobalState; use crate::global_timer::GlobalTimer; use crate::http_util::create_http_client; use crate::import_map::ImportMap; use crate::metrics::Metrics; use crate::ops::serialize_result; use crate::ops::JsonOp; use crate::ops::MinimalOp; use crate::permissions::Permissions; use crate::tsc::TargetLib; use crate::web_worker::WebWorkerHandle; use deno_core::Buf; use deno_core::BufVec; use deno_core::CoreIsolateState; use deno_core::ErrBox; use deno_core::ModuleLoadId; use deno_core::ModuleLoader; use deno_core::ModuleSpecifier; use deno_core::Op; use deno_core::ResourceTable; use deno_core::ZeroCopyBuf; use futures::future::FutureExt; use futures::Future; use rand::rngs::StdRng; use rand::SeedableRng; use serde_json::Value; use std::cell::Cell; use std::cell::RefCell; use std::collections::HashMap; use std::path::Path; use std::pin::Pin; use std::rc::Rc; use std::str; use std::sync::Arc; use std::thread::JoinHandle; use std::time::Instant; #[cfg_attr(feature = "cargo-clippy", allow(stutter))] pub struct State { pub global_state: Arc, pub permissions: RefCell, pub main_module: ModuleSpecifier, /// When flags contains a `.import_map_path` option, the content of the /// import map file will be resolved and set. pub import_map: Option, pub metrics: RefCell, pub global_timer: RefCell, pub workers: RefCell, WebWorkerHandle)>>, pub next_worker_id: Cell, pub start_time: Instant, pub seeded_rng: Option>, pub target_lib: TargetLib, pub is_main: bool, pub is_internal: bool, pub http_client: RefCell, } impl State { pub fn stateful_json_op_sync( self: &Rc, resource_table: &Rc>, dispatcher: D, ) -> impl Fn(&mut deno_core::CoreIsolateState, &mut [ZeroCopyBuf]) -> Op where D: Fn( &State, &mut ResourceTable, Value, &mut [ZeroCopyBuf], ) -> Result, { let state = self.clone(); let resource_table = resource_table.clone(); let f = move |isolate_state: &mut CoreIsolateState, bufs: &mut [ZeroCopyBuf]| { let get_error_class_fn = isolate_state.get_error_class_fn; // The first buffer should contain JSON encoded op arguments; parse them. let args: Value = match serde_json::from_slice(&bufs[0]) { Ok(v) => v, Err(e) => { return Op::Sync(serialize_result( None, Err(e.into()), get_error_class_fn, )); } }; // Make a slice containing all buffers except for the first one. let zero_copy = &mut bufs[1..]; let result = dispatcher(&state, &mut *resource_table.borrow_mut(), args, zero_copy); // Convert to Op. Op::Sync(serialize_result(None, result, get_error_class_fn)) }; self.core_op(f) } pub fn stateful_json_op_async( self: &Rc, resource_table: &Rc>, dispatcher: D, ) -> impl Fn(&mut CoreIsolateState, &mut [ZeroCopyBuf]) -> Op where D: FnOnce(Rc, Rc>, Value, BufVec) -> F + Clone, F: Future> + 'static, { let state = self.clone(); let resource_table = resource_table.clone(); let f = move |isolate_state: &mut CoreIsolateState, bufs: &mut [ZeroCopyBuf]| { let get_error_class_fn = isolate_state.get_error_class_fn; // The first buffer should contain JSON encoded op arguments; parse them. let args: Value = match serde_json::from_slice(&bufs[0]) { Ok(v) => v, Err(e) => { let e = e.into(); return Op::Sync(serialize_result(None, Err(e), get_error_class_fn)); } }; // `args` should have a `promiseId` property with positive integer value. let promise_id = match args.get("promiseId").and_then(|v| v.as_u64()) { Some(i) => i, None => { let e = ErrBox::new("TypeError", "`promiseId` invalid/missing"); return Op::Sync(serialize_result(None, Err(e), get_error_class_fn)); } }; // Take ownership of all buffers after the first one. let zero_copy: BufVec = bufs[1..].into(); // Call dispatcher to obtain op future. let fut = (dispatcher.clone())( state.clone(), resource_table.clone(), args, zero_copy, ); // Convert to Op. Op::Async( async move { serialize_result(Some(promise_id), fut.await, get_error_class_fn) } .boxed_local(), ) }; self.core_op(f) } // TODO(bartlomieju): remove me - still used by `op_open_plugin` which // needs access to isolate_state pub fn stateful_json_op2( self: &Rc, dispatcher: D, ) -> impl Fn(&mut deno_core::CoreIsolateState, &mut [ZeroCopyBuf]) -> Op where D: Fn( &mut deno_core::CoreIsolateState, &Rc, Value, &mut [ZeroCopyBuf], ) -> Result, { use crate::ops::json_op; self.core_op(json_op(self.stateful_op2(dispatcher))) } /// Wrap core `OpDispatcher` to collect metrics. // TODO(ry) this should be private. Is called by stateful_json_op or // stateful_minimal_op pub(crate) fn core_op( self: &Rc, dispatcher: D, ) -> impl Fn(&mut deno_core::CoreIsolateState, &mut [ZeroCopyBuf]) -> Op where D: Fn(&mut deno_core::CoreIsolateState, &mut [ZeroCopyBuf]) -> Op, { let state = self.clone(); move |isolate_state: &mut deno_core::CoreIsolateState, zero_copy: &mut [ZeroCopyBuf]| -> Op { let bytes_sent_control = zero_copy.get(0).map(|s| s.len()).unwrap_or(0) as u64; let bytes_sent_zero_copy = zero_copy[1..].iter().map(|b| b.len()).sum::() as u64; let op = dispatcher(isolate_state, zero_copy); match op { Op::Sync(buf) => { state.metrics.borrow_mut().op_sync( bytes_sent_control, bytes_sent_zero_copy, buf.len() as u64, ); Op::Sync(buf) } Op::Async(fut) => { state .metrics .borrow_mut() .op_dispatched_async(bytes_sent_control, bytes_sent_zero_copy); let state = state.clone(); let result_fut = fut.map(move |buf: Buf| { state .metrics .borrow_mut() .op_completed_async(buf.len() as u64); buf }); Op::Async(result_fut.boxed_local()) } Op::AsyncUnref(fut) => { state.metrics.borrow_mut().op_dispatched_async_unref( bytes_sent_control, bytes_sent_zero_copy, ); let state = state.clone(); let result_fut = fut.map(move |buf: Buf| { state .metrics .borrow_mut() .op_completed_async_unref(buf.len() as u64); buf }); Op::AsyncUnref(result_fut.boxed_local()) } } } } pub fn stateful_minimal_op2( self: &Rc, dispatcher: D, ) -> impl Fn(&mut deno_core::CoreIsolateState, &mut [ZeroCopyBuf]) -> Op where D: Fn( &mut deno_core::CoreIsolateState, &Rc, bool, i32, &mut [ZeroCopyBuf], ) -> MinimalOp, { let state = self.clone(); self.core_op(crate::ops::minimal_op( move |isolate_state: &mut deno_core::CoreIsolateState, is_sync: bool, rid: i32, zero_copy: &mut [ZeroCopyBuf]| -> MinimalOp { dispatcher(isolate_state, &state, is_sync, rid, zero_copy) }, )) } /// This is a special function that provides `state` argument to dispatcher. /// /// NOTE: This only works with JSON dispatcher. /// This is a band-aid for transition to `CoreIsolate.register_op` API as most of our /// ops require `state` argument. pub fn stateful_op( self: &Rc, dispatcher: D, ) -> impl Fn( &mut deno_core::CoreIsolateState, Value, &mut [ZeroCopyBuf], ) -> Result where D: Fn(&Rc, Value, &mut [ZeroCopyBuf]) -> Result, { let state = self.clone(); move |_isolate_state: &mut deno_core::CoreIsolateState, args: Value, zero_copy: &mut [ZeroCopyBuf]| -> Result { dispatcher(&state, args, zero_copy) } } pub fn stateful_op2( self: &Rc, dispatcher: D, ) -> impl Fn( &mut deno_core::CoreIsolateState, Value, &mut [ZeroCopyBuf], ) -> Result where D: Fn( &mut deno_core::CoreIsolateState, &Rc, Value, &mut [ZeroCopyBuf], ) -> Result, { let state = self.clone(); move |isolate_state: &mut deno_core::CoreIsolateState, args: Value, zero_copy: &mut [ZeroCopyBuf]| -> Result { dispatcher(isolate_state, &state, args, zero_copy) } } /// Quits the process if the --unstable flag was not provided. /// /// This is intentionally a non-recoverable check so that people cannot probe /// for unstable APIs from stable programs. pub fn check_unstable(&self, api_name: &str) { // TODO(ry) Maybe use IsolateHandle::terminate_execution here to provide a // stack trace in JS. if !self.global_state.flags.unstable { exit_unstable(api_name); } } } pub fn exit_unstable(api_name: &str) { eprintln!( "Unstable API '{}'. The --unstable flag must be provided.", api_name ); std::process::exit(70); } impl ModuleLoader for State { fn resolve( &self, specifier: &str, referrer: &str, is_main: bool, ) -> Result { if !is_main { if let Some(import_map) = &self.import_map { let result = import_map.resolve(specifier, referrer)?; if let Some(r) = result { return Ok(r); } } } let module_specifier = ModuleSpecifier::resolve_import(specifier, referrer)?; Ok(module_specifier) } fn load( &self, module_specifier: &ModuleSpecifier, maybe_referrer: Option, _is_dyn_import: bool, ) -> Pin> { let module_specifier = module_specifier.to_owned(); // TODO(bartlomieju): incrementing resolve_count here has no sense... self.metrics.borrow_mut().resolve_count += 1; let module_url_specified = module_specifier.to_string(); let global_state = self.global_state.clone(); // TODO(bartlomieju): `fetch_compiled_module` should take `load_id` param let fut = async move { let compiled_module = global_state .fetch_compiled_module(module_specifier, maybe_referrer) .await?; Ok(deno_core::ModuleSource { // Real module name, might be different from initial specifier // due to redirections. code: compiled_module.code, module_url_specified, module_url_found: compiled_module.name, }) }; fut.boxed_local() } fn prepare_load( &self, _load_id: ModuleLoadId, module_specifier: &ModuleSpecifier, maybe_referrer: Option, is_dyn_import: bool, ) -> Pin>>> { let module_specifier = module_specifier.clone(); let target_lib = self.target_lib.clone(); let maybe_import_map = self.import_map.clone(); // Only "main" module is loaded without permission check, // ie. module that is associated with "is_main" state // and is not a dynamic import. let permissions = if self.is_main && !is_dyn_import { Permissions::allow_all() } else { self.permissions.borrow().clone() }; let global_state = self.global_state.clone(); // TODO(bartlomieju): I'm not sure if it's correct to ignore // bad referrer - this is the case for `Deno.core.evalContext()` where // `ref_str` is ``. let maybe_referrer = if let Some(ref_str) = maybe_referrer { ModuleSpecifier::resolve_url(&ref_str).ok() } else { None }; // TODO(bartlomieju): `prepare_module_load` should take `load_id` param async move { global_state .prepare_module_load( module_specifier, maybe_referrer, target_lib, permissions, is_dyn_import, maybe_import_map, ) .await } .boxed_local() } } impl State { /// If `shared_permission` is None then permissions from globa state are used. pub fn new( global_state: &Arc, shared_permissions: Option, main_module: ModuleSpecifier, maybe_import_map: Option, is_internal: bool, ) -> Result, ErrBox> { let fl = &global_state.flags; let state = State { global_state: global_state.clone(), main_module, permissions: shared_permissions .unwrap_or_else(|| global_state.permissions.clone()) .into(), import_map: maybe_import_map, metrics: Default::default(), global_timer: Default::default(), workers: Default::default(), next_worker_id: Default::default(), start_time: Instant::now(), seeded_rng: fl.seed.map(|v| StdRng::seed_from_u64(v).into()), target_lib: TargetLib::Main, is_main: true, is_internal, http_client: create_http_client(fl.ca_file.as_deref())?.into(), }; Ok(Rc::new(state)) } /// If `shared_permission` is None then permissions from globa state are used. pub fn new_for_worker( global_state: &Arc, shared_permissions: Option, main_module: ModuleSpecifier, ) -> Result, ErrBox> { let fl = &global_state.flags; let state = State { global_state: global_state.clone(), main_module, permissions: shared_permissions .unwrap_or_else(|| global_state.permissions.clone()) .into(), import_map: None, metrics: Default::default(), global_timer: Default::default(), workers: Default::default(), next_worker_id: Default::default(), start_time: Instant::now(), seeded_rng: fl.seed.map(|v| StdRng::seed_from_u64(v).into()), target_lib: TargetLib::Worker, is_main: false, is_internal: false, http_client: create_http_client(fl.ca_file.as_deref())?.into(), }; Ok(Rc::new(state)) } #[inline] pub fn check_read(&self, path: &Path) -> Result<(), ErrBox> { self.permissions.borrow().check_read(path) } /// As `check_read()`, but permission error messages will anonymize the path /// by replacing it with the given `display`. #[inline] pub fn check_read_blind( &self, path: &Path, display: &str, ) -> Result<(), ErrBox> { self.permissions.borrow().check_read_blind(path, display) } #[inline] pub fn check_write(&self, path: &Path) -> Result<(), ErrBox> { self.permissions.borrow().check_write(path) } #[inline] pub fn check_env(&self) -> Result<(), ErrBox> { self.permissions.borrow().check_env() } #[inline] pub fn check_net(&self, hostname: &str, port: u16) -> Result<(), ErrBox> { self.permissions.borrow().check_net(hostname, port) } #[inline] pub fn check_net_url(&self, url: &url::Url) -> Result<(), ErrBox> { self.permissions.borrow().check_net_url(url) } #[inline] pub fn check_run(&self) -> Result<(), ErrBox> { self.permissions.borrow().check_run() } #[inline] pub fn check_hrtime(&self) -> Result<(), ErrBox> { self.permissions.borrow().check_hrtime() } #[inline] pub fn check_plugin(&self, filename: &Path) -> Result<(), ErrBox> { self.permissions.borrow().check_plugin(filename) } pub fn check_dyn_import( &self, module_specifier: &ModuleSpecifier, ) -> Result<(), ErrBox> { let u = module_specifier.as_url(); // TODO(bartlomieju): temporary fix to prevent hitting `unreachable` // statement that is actually reachable... SourceFileFetcher::check_if_supported_scheme(u)?; match u.scheme() { "http" | "https" => { self.check_net_url(u)?; Ok(()) } "file" => { let path = u .to_file_path() .unwrap() .into_os_string() .into_string() .unwrap(); self.check_read(Path::new(&path))?; Ok(()) } _ => unreachable!(), } } #[cfg(test)] pub fn mock(main_module: &str) -> Rc { let module_specifier = ModuleSpecifier::resolve_url_or_path(main_module) .expect("Invalid entry module"); State::new( &GlobalState::mock(vec!["deno".to_string()], None), None, module_specifier, None, false, ) .unwrap() } }