2023-01-02 16:00:42 -05:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
2022-08-09 21:06:01 +02:00
|
|
|
|
2022-11-08 03:16:24 +01:00
|
|
|
use deno_core::error::generic_error;
|
2022-08-09 21:06:01 +02:00
|
|
|
use deno_core::error::AnyError;
|
|
|
|
use deno_core::include_js_files;
|
|
|
|
use deno_core::normalize_path;
|
|
|
|
use deno_core::op;
|
2022-08-20 11:31:33 -04:00
|
|
|
use deno_core::url::Url;
|
2022-08-09 21:06:01 +02:00
|
|
|
use deno_core::Extension;
|
2022-11-28 21:59:36 +01:00
|
|
|
use deno_core::JsRuntimeInspector;
|
2022-08-09 22:09:51 -04:00
|
|
|
use deno_core::OpState;
|
2022-09-08 22:01:48 +02:00
|
|
|
use once_cell::sync::Lazy;
|
2022-09-14 11:59:20 -04:00
|
|
|
use std::collections::HashSet;
|
2022-08-20 11:31:33 -04:00
|
|
|
use std::path::Path;
|
2022-08-09 21:06:01 +02:00
|
|
|
use std::path::PathBuf;
|
2022-08-20 11:31:33 -04:00
|
|
|
use std::rc::Rc;
|
|
|
|
|
2022-09-05 12:36:35 +02:00
|
|
|
pub mod errors;
|
|
|
|
mod package_json;
|
2022-09-22 11:17:02 -04:00
|
|
|
mod path;
|
2022-09-05 12:36:35 +02:00
|
|
|
mod resolution;
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
pub use package_json::PackageJson;
|
2022-09-22 11:17:02 -04:00
|
|
|
pub use path::PathClean;
|
2022-08-30 14:09:22 -04:00
|
|
|
pub use resolution::get_closest_package_json;
|
2022-08-20 11:31:33 -04:00
|
|
|
pub use resolution::get_package_scope_config;
|
|
|
|
pub use resolution::legacy_main_resolve;
|
|
|
|
pub use resolution::package_exports_resolve;
|
|
|
|
pub use resolution::package_imports_resolve;
|
|
|
|
pub use resolution::package_resolve;
|
2022-11-16 20:41:27 +01:00
|
|
|
pub use resolution::path_to_declaration_path;
|
2022-08-30 14:09:22 -04:00
|
|
|
pub use resolution::NodeModuleKind;
|
2022-11-30 18:07:32 -05:00
|
|
|
pub use resolution::NodeResolutionMode;
|
2022-08-20 11:31:33 -04:00
|
|
|
pub use resolution::DEFAULT_CONDITIONS;
|
2022-11-28 21:59:36 +01:00
|
|
|
use std::cell::RefCell;
|
2022-08-20 11:31:33 -04:00
|
|
|
|
2022-08-24 18:07:49 +02:00
|
|
|
pub trait NodePermissions {
|
|
|
|
fn check_read(&mut self, path: &Path) -> Result<(), AnyError>;
|
|
|
|
}
|
|
|
|
|
2022-09-13 11:59:01 -04:00
|
|
|
pub trait RequireNpmResolver {
|
2022-08-20 11:31:33 -04:00
|
|
|
fn resolve_package_folder_from_package(
|
|
|
|
&self,
|
|
|
|
specifier: &str,
|
|
|
|
referrer: &Path,
|
2022-11-30 18:07:32 -05:00
|
|
|
mode: NodeResolutionMode,
|
2022-08-20 11:31:33 -04:00
|
|
|
) -> Result<PathBuf, AnyError>;
|
|
|
|
|
|
|
|
fn resolve_package_folder_from_path(
|
|
|
|
&self,
|
|
|
|
path: &Path,
|
|
|
|
) -> Result<PathBuf, AnyError>;
|
|
|
|
|
|
|
|
fn in_npm_package(&self, path: &Path) -> bool;
|
|
|
|
|
2023-01-10 14:35:44 +01:00
|
|
|
fn ensure_read_permission(
|
|
|
|
&self,
|
|
|
|
permissions: &mut dyn NodePermissions,
|
|
|
|
path: &Path,
|
|
|
|
) -> Result<(), AnyError>;
|
2022-08-20 11:31:33 -04:00
|
|
|
}
|
|
|
|
|
2022-08-24 14:54:20 +02:00
|
|
|
pub const MODULE_ES_SHIM: &str = include_str!("./module_es_shim.js");
|
|
|
|
|
2022-09-08 22:01:48 +02:00
|
|
|
pub static NODE_GLOBAL_THIS_NAME: Lazy<String> = Lazy::new(|| {
|
|
|
|
let now = std::time::SystemTime::now();
|
|
|
|
let seconds = now
|
|
|
|
.duration_since(std::time::SystemTime::UNIX_EPOCH)
|
|
|
|
.unwrap()
|
|
|
|
.as_secs();
|
|
|
|
// use a changing variable name to make it hard to depend on this
|
2023-01-27 10:43:16 -05:00
|
|
|
format!("__DENO_NODE_GLOBAL_THIS_{seconds}__")
|
2022-09-08 22:01:48 +02:00
|
|
|
});
|
|
|
|
|
2022-09-14 11:59:20 -04:00
|
|
|
pub static NODE_ENV_VAR_ALLOWLIST: Lazy<HashSet<String>> = Lazy::new(|| {
|
|
|
|
// The full list of environment variables supported by Node.js is available
|
|
|
|
// at https://nodejs.org/api/cli.html#environment-variables
|
|
|
|
let mut set = HashSet::new();
|
|
|
|
set.insert("NODE_DEBUG".to_string());
|
|
|
|
set.insert("NODE_OPTIONS".to_string());
|
|
|
|
set
|
|
|
|
});
|
|
|
|
|
2022-08-24 18:07:49 +02:00
|
|
|
pub fn init<P: NodePermissions + 'static>(
|
2022-09-13 11:59:01 -04:00
|
|
|
maybe_npm_resolver: Option<Rc<dyn RequireNpmResolver>>,
|
2022-08-20 11:31:33 -04:00
|
|
|
) -> Extension {
|
2023-01-08 23:48:46 +01:00
|
|
|
Extension::builder(env!("CARGO_PKG_NAME"))
|
2023-02-07 22:09:50 +01:00
|
|
|
.esm(include_js_files!("01_node.js", "02_require.js",))
|
2022-08-09 21:06:01 +02:00
|
|
|
.ops(vec![
|
|
|
|
op_require_init_paths::decl(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_node_module_paths::decl::<P>(),
|
2022-08-09 21:06:01 +02:00
|
|
|
op_require_proxy_path::decl(),
|
2022-08-20 11:31:33 -04:00
|
|
|
op_require_is_deno_dir_package::decl(),
|
|
|
|
op_require_resolve_deno_dir::decl(),
|
2022-08-09 21:06:01 +02:00
|
|
|
op_require_is_request_relative::decl(),
|
|
|
|
op_require_resolve_lookup_paths::decl(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_try_self_parent_path::decl::<P>(),
|
2023-01-10 14:35:44 +01:00
|
|
|
op_require_try_self::decl::<P>(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_real_path::decl::<P>(),
|
2022-08-09 21:06:01 +02:00
|
|
|
op_require_path_is_absolute::decl(),
|
|
|
|
op_require_path_dirname::decl(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_stat::decl::<P>(),
|
2022-08-09 21:06:01 +02:00
|
|
|
op_require_path_resolve::decl(),
|
|
|
|
op_require_path_basename::decl(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_read_file::decl::<P>(),
|
2022-08-20 11:31:33 -04:00
|
|
|
op_require_as_file_path::decl(),
|
2023-01-10 14:35:44 +01:00
|
|
|
op_require_resolve_exports::decl::<P>(),
|
2022-08-30 14:09:22 -04:00
|
|
|
op_require_read_closest_package_json::decl::<P>(),
|
2023-01-10 14:35:44 +01:00
|
|
|
op_require_read_package_scope::decl::<P>(),
|
2022-08-24 18:07:49 +02:00
|
|
|
op_require_package_imports_resolve::decl::<P>(),
|
2022-11-28 21:59:36 +01:00
|
|
|
op_require_break_on_next_statement::decl(),
|
2022-08-09 21:06:01 +02:00
|
|
|
])
|
2022-08-09 22:09:51 -04:00
|
|
|
.state(move |state| {
|
2022-08-20 11:31:33 -04:00
|
|
|
if let Some(npm_resolver) = maybe_npm_resolver.clone() {
|
|
|
|
state.put(npm_resolver);
|
|
|
|
}
|
2022-08-09 22:09:51 -04:00
|
|
|
Ok(())
|
|
|
|
})
|
2022-08-09 21:06:01 +02:00
|
|
|
.build()
|
|
|
|
}
|
|
|
|
|
2022-08-24 18:07:49 +02:00
|
|
|
fn ensure_read_permission<P>(
|
2022-08-20 11:31:33 -04:00
|
|
|
state: &mut OpState,
|
|
|
|
file_path: &Path,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<(), AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
let resolver = {
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>();
|
2022-08-20 11:31:33 -04:00
|
|
|
resolver.clone()
|
|
|
|
};
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
|
|
|
resolver.ensure_read_permission(permissions, file_path)
|
2022-08-20 11:31:33 -04:00
|
|
|
}
|
|
|
|
|
2022-08-09 21:06:01 +02:00
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
pub fn op_require_init_paths() -> Vec<String> {
|
2022-08-20 11:31:33 -04:00
|
|
|
// todo(dsherret): this code is node compat mode specific and
|
|
|
|
// we probably don't want it for small mammal, so ignore it for now
|
|
|
|
|
|
|
|
// let (home_dir, node_path) = if cfg!(windows) {
|
|
|
|
// (
|
|
|
|
// std::env::var("USERPROFILE").unwrap_or_else(|_| "".into()),
|
|
|
|
// std::env::var("NODE_PATH").unwrap_or_else(|_| "".into()),
|
|
|
|
// )
|
|
|
|
// } else {
|
|
|
|
// (
|
|
|
|
// std::env::var("HOME").unwrap_or_else(|_| "".into()),
|
|
|
|
// std::env::var("NODE_PATH").unwrap_or_else(|_| "".into()),
|
|
|
|
// )
|
|
|
|
// };
|
|
|
|
|
|
|
|
// let mut prefix_dir = std::env::current_exe().unwrap();
|
|
|
|
// if cfg!(windows) {
|
|
|
|
// prefix_dir = prefix_dir.join("..").join("..")
|
|
|
|
// } else {
|
|
|
|
// prefix_dir = prefix_dir.join("..")
|
|
|
|
// }
|
2022-08-09 21:06:01 +02:00
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
// let mut paths = vec![prefix_dir.join("lib").join("node")];
|
2022-08-09 21:06:01 +02:00
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
// if !home_dir.is_empty() {
|
|
|
|
// paths.insert(0, PathBuf::from(&home_dir).join(".node_libraries"));
|
|
|
|
// paths.insert(0, PathBuf::from(&home_dir).join(".nod_modules"));
|
|
|
|
// }
|
2022-08-09 21:06:01 +02:00
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
// let mut paths = paths
|
|
|
|
// .into_iter()
|
|
|
|
// .map(|p| p.to_string_lossy().to_string())
|
|
|
|
// .collect();
|
|
|
|
|
|
|
|
// if !node_path.is_empty() {
|
|
|
|
// let delimiter = if cfg!(windows) { ";" } else { ":" };
|
|
|
|
// let mut node_paths: Vec<String> = node_path
|
|
|
|
// .split(delimiter)
|
|
|
|
// .filter(|e| !e.is_empty())
|
|
|
|
// .map(|s| s.to_string())
|
|
|
|
// .collect();
|
|
|
|
// node_paths.append(&mut paths);
|
|
|
|
// paths = node_paths;
|
|
|
|
// }
|
2022-08-09 21:06:01 +02:00
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
vec![]
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
pub fn op_require_node_module_paths<P>(
|
2022-08-09 22:09:51 -04:00
|
|
|
state: &mut OpState,
|
|
|
|
from: String,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<Vec<String>, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-09 21:06:01 +02:00
|
|
|
// Guarantee that "from" is absolute.
|
|
|
|
let from = deno_core::resolve_path(&from)
|
|
|
|
.unwrap()
|
|
|
|
.to_file_path()
|
|
|
|
.unwrap();
|
|
|
|
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &from)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
|
2022-08-09 21:06:01 +02:00
|
|
|
if cfg!(windows) {
|
|
|
|
// return root node_modules when path is 'D:\\'.
|
|
|
|
let from_str = from.to_str().unwrap();
|
|
|
|
if from_str.len() >= 3 {
|
|
|
|
let bytes = from_str.as_bytes();
|
|
|
|
if bytes[from_str.len() - 1] == b'\\' && bytes[from_str.len() - 2] == b':'
|
|
|
|
{
|
|
|
|
let p = from_str.to_owned() + "node_modules";
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(vec![p]);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Return early not only to avoid unnecessary work, but to *avoid* returning
|
|
|
|
// an array of two items for a root: [ '//node_modules', '/node_modules' ]
|
|
|
|
if from.to_string_lossy() == "/" {
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(vec!["/node_modules".to_string()]);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut paths = vec![];
|
|
|
|
let mut current_path = from.as_path();
|
|
|
|
let mut maybe_parent = Some(current_path);
|
|
|
|
while let Some(parent) = maybe_parent {
|
|
|
|
if !parent.ends_with("/node_modules") {
|
|
|
|
paths.push(parent.join("node_modules").to_string_lossy().to_string());
|
|
|
|
current_path = parent;
|
|
|
|
maybe_parent = current_path.parent();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !cfg!(windows) {
|
|
|
|
// Append /node_modules to handle root paths.
|
|
|
|
paths.push("/node_modules".to_string());
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
Ok(paths)
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
fn op_require_proxy_path(filename: String) -> String {
|
2022-08-09 21:06:01 +02:00
|
|
|
// Allow a directory to be passed as the filename
|
|
|
|
let trailing_slash = if cfg!(windows) {
|
2022-11-29 14:13:14 +09:00
|
|
|
// Node also counts a trailing forward slash as a
|
|
|
|
// directory for node on Windows, but not backslashes
|
|
|
|
// on non-Windows platforms
|
|
|
|
filename.ends_with('\\') || filename.ends_with('/')
|
2022-08-09 21:06:01 +02:00
|
|
|
} else {
|
|
|
|
filename.ends_with('/')
|
|
|
|
};
|
|
|
|
|
|
|
|
if trailing_slash {
|
|
|
|
let p = PathBuf::from(filename);
|
|
|
|
p.join("noop.js").to_string_lossy().to_string()
|
|
|
|
} else {
|
|
|
|
filename
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-12-15 16:26:10 +01:00
|
|
|
fn op_require_is_request_relative(request: String) -> bool {
|
2022-08-20 11:31:33 -04:00
|
|
|
if request.starts_with("./") || request.starts_with("../") || request == ".."
|
|
|
|
{
|
2022-08-09 21:06:01 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if cfg!(windows) {
|
|
|
|
if request.starts_with(".\\") {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if request.starts_with("..\\") {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
#[op]
|
|
|
|
fn op_require_resolve_deno_dir(
|
|
|
|
state: &mut OpState,
|
|
|
|
request: String,
|
|
|
|
parent_filename: String,
|
|
|
|
) -> Option<String> {
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>();
|
2022-08-20 11:31:33 -04:00
|
|
|
resolver
|
|
|
|
.resolve_package_folder_from_package(
|
|
|
|
&request,
|
|
|
|
&PathBuf::from(parent_filename),
|
2022-11-30 18:07:32 -05:00
|
|
|
NodeResolutionMode::Execution,
|
2022-08-20 11:31:33 -04:00
|
|
|
)
|
|
|
|
.ok()
|
|
|
|
.map(|p| p.to_string_lossy().to_string())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
|
|
|
fn op_require_is_deno_dir_package(state: &mut OpState, path: String) -> bool {
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>();
|
2022-08-20 11:31:33 -04:00
|
|
|
resolver.in_npm_package(&PathBuf::from(path))
|
|
|
|
}
|
|
|
|
|
2022-08-09 21:06:01 +02:00
|
|
|
#[op]
|
|
|
|
fn op_require_resolve_lookup_paths(
|
|
|
|
request: String,
|
|
|
|
maybe_parent_paths: Option<Vec<String>>,
|
|
|
|
parent_filename: String,
|
|
|
|
) -> Option<Vec<String>> {
|
|
|
|
if !request.starts_with('.')
|
|
|
|
|| (request.len() > 1
|
|
|
|
&& !request.starts_with("..")
|
|
|
|
&& !request.starts_with("./")
|
|
|
|
&& (!cfg!(windows) || !request.starts_with(".\\")))
|
|
|
|
{
|
|
|
|
let module_paths = vec![];
|
|
|
|
let mut paths = module_paths;
|
|
|
|
if let Some(mut parent_paths) = maybe_parent_paths {
|
|
|
|
if !parent_paths.is_empty() {
|
|
|
|
paths.append(&mut parent_paths);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !paths.is_empty() {
|
|
|
|
return Some(paths);
|
|
|
|
} else {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// In REPL, parent.filename is null.
|
|
|
|
// if (!parent || !parent.id || !parent.filename) {
|
|
|
|
// // Make require('./path/to/foo') work - normally the path is taken
|
|
|
|
// // from realpath(__filename) but in REPL there is no filename
|
|
|
|
// const mainPaths = ['.'];
|
|
|
|
|
|
|
|
// debug('looking for %j in %j', request, mainPaths);
|
|
|
|
// return mainPaths;
|
|
|
|
// }
|
|
|
|
|
|
|
|
let p = PathBuf::from(parent_filename);
|
|
|
|
Some(vec![p.parent().unwrap().to_string_lossy().to_string()])
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
fn op_require_path_is_absolute(p: String) -> bool {
|
2022-08-09 21:06:01 +02:00
|
|
|
PathBuf::from(p).is_absolute()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
fn op_require_stat<P>(
|
|
|
|
state: &mut OpState,
|
|
|
|
path: String,
|
|
|
|
) -> Result<i32, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
let path = PathBuf::from(path);
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &path)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
if let Ok(metadata) = std::fs::metadata(&path) {
|
2022-08-09 21:06:01 +02:00
|
|
|
if metadata.is_file() {
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(0);
|
2022-08-09 21:06:01 +02:00
|
|
|
} else {
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(1);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
Ok(-1)
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
fn op_require_real_path<P>(
|
2022-08-09 22:09:51 -04:00
|
|
|
state: &mut OpState,
|
|
|
|
request: String,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<String, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
let path = PathBuf::from(request);
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &path)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
let mut canonicalized_path = path.canonicalize()?;
|
2022-08-09 21:06:01 +02:00
|
|
|
if cfg!(windows) {
|
|
|
|
canonicalized_path = PathBuf::from(
|
|
|
|
canonicalized_path
|
|
|
|
.display()
|
|
|
|
.to_string()
|
|
|
|
.trim_start_matches("\\\\?\\"),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(canonicalized_path.to_string_lossy().to_string())
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
fn path_resolve(parts: Vec<String>) -> String {
|
2022-08-09 21:06:01 +02:00
|
|
|
assert!(!parts.is_empty());
|
|
|
|
let mut p = PathBuf::from(&parts[0]);
|
|
|
|
if parts.len() > 1 {
|
|
|
|
for part in &parts[1..] {
|
|
|
|
p = p.join(part);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
normalize_path(p).to_string_lossy().to_string()
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
fn op_require_path_resolve(parts: Vec<String>) -> String {
|
2022-08-20 11:31:33 -04:00
|
|
|
path_resolve(parts)
|
|
|
|
}
|
|
|
|
|
2022-08-09 21:06:01 +02:00
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
fn op_require_path_dirname(request: String) -> Result<String, AnyError> {
|
2022-08-09 21:06:01 +02:00
|
|
|
let p = PathBuf::from(request);
|
2022-11-08 03:16:24 +01:00
|
|
|
if let Some(parent) = p.parent() {
|
|
|
|
Ok(parent.to_string_lossy().to_string())
|
|
|
|
} else {
|
|
|
|
Err(generic_error("Path doesn't have a parent"))
|
|
|
|
}
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
fn op_require_path_basename(request: String) -> Result<String, AnyError> {
|
2022-08-09 21:06:01 +02:00
|
|
|
let p = PathBuf::from(request);
|
2022-11-08 03:16:24 +01:00
|
|
|
if let Some(path) = p.file_name() {
|
|
|
|
Ok(path.to_string_lossy().to_string())
|
|
|
|
} else {
|
|
|
|
Err(generic_error("Path doesn't have a file name"))
|
|
|
|
}
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
fn op_require_try_self_parent_path<P>(
|
2022-08-09 22:09:51 -04:00
|
|
|
state: &mut OpState,
|
2022-08-09 21:06:01 +02:00
|
|
|
has_parent: bool,
|
|
|
|
maybe_parent_filename: Option<String>,
|
|
|
|
maybe_parent_id: Option<String>,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<Option<String>, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-09 21:06:01 +02:00
|
|
|
if !has_parent {
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(None);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(parent_filename) = maybe_parent_filename {
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(Some(parent_filename));
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(parent_id) = maybe_parent_id {
|
|
|
|
if parent_id == "<repl>" || parent_id == "internal/preload" {
|
|
|
|
if let Ok(cwd) = std::env::current_dir() {
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &cwd)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
return Ok(Some(cwd.to_string_lossy().to_string()));
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-20 11:31:33 -04:00
|
|
|
Ok(None)
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2023-01-10 14:35:44 +01:00
|
|
|
fn op_require_try_self<P>(
|
2022-08-09 22:09:51 -04:00
|
|
|
state: &mut OpState,
|
2022-08-20 11:31:33 -04:00
|
|
|
parent_path: Option<String>,
|
|
|
|
request: String,
|
2023-01-10 14:35:44 +01:00
|
|
|
) -> Result<Option<String>, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
if parent_path.is_none() {
|
|
|
|
return Ok(None);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>().clone();
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
2022-08-20 11:31:33 -04:00
|
|
|
let pkg = resolution::get_package_scope_config(
|
|
|
|
&Url::from_file_path(parent_path.unwrap()).unwrap(),
|
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-20 11:31:33 -04:00
|
|
|
)
|
|
|
|
.ok();
|
|
|
|
if pkg.is_none() {
|
|
|
|
return Ok(None);
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
let pkg = pkg.unwrap();
|
|
|
|
if pkg.exports.is_none() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
if pkg.name.is_none() {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
|
|
|
|
let pkg_name = pkg.name.as_ref().unwrap().to_string();
|
|
|
|
let mut expansion = ".".to_string();
|
|
|
|
|
|
|
|
if request == pkg_name {
|
|
|
|
// pass
|
2023-01-27 10:43:16 -05:00
|
|
|
} else if request.starts_with(&format!("{pkg_name}/")) {
|
2022-08-20 11:31:33 -04:00
|
|
|
expansion += &request[pkg_name.len()..];
|
|
|
|
} else {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
|
2022-08-30 14:09:22 -04:00
|
|
|
let referrer = deno_core::url::Url::from_file_path(&pkg.path).unwrap();
|
2022-08-20 11:31:33 -04:00
|
|
|
if let Some(exports) = &pkg.exports {
|
|
|
|
resolution::package_exports_resolve(
|
2022-08-30 14:09:22 -04:00
|
|
|
&pkg.path,
|
2022-08-20 11:31:33 -04:00
|
|
|
expansion,
|
|
|
|
exports,
|
2022-08-30 14:09:22 -04:00
|
|
|
&referrer,
|
|
|
|
NodeModuleKind::Cjs,
|
2022-08-20 11:31:33 -04:00
|
|
|
resolution::REQUIRE_CONDITIONS,
|
2022-11-30 18:07:32 -05:00
|
|
|
NodeResolutionMode::Execution,
|
2022-08-20 11:31:33 -04:00
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-20 11:31:33 -04:00
|
|
|
)
|
2022-08-30 14:09:22 -04:00
|
|
|
.map(|r| Some(r.to_string_lossy().to_string()))
|
2022-08-20 11:31:33 -04:00
|
|
|
} else {
|
|
|
|
Ok(None)
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
fn op_require_read_file<P>(
|
2022-08-09 22:09:51 -04:00
|
|
|
state: &mut OpState,
|
2022-08-20 11:31:33 -04:00
|
|
|
file_path: String,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<String, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
let file_path = PathBuf::from(file_path);
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &file_path)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
Ok(std::fs::read_to_string(file_path)?)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-11-10 17:57:10 +01:00
|
|
|
pub fn op_require_as_file_path(file_or_url: String) -> String {
|
2022-11-08 03:16:24 +01:00
|
|
|
if let Ok(url) = Url::parse(&file_or_url) {
|
|
|
|
if let Ok(p) = url.to_file_path() {
|
|
|
|
return p.to_string_lossy().to_string();
|
|
|
|
}
|
2022-08-20 11:31:33 -04:00
|
|
|
}
|
2022-11-08 03:16:24 +01:00
|
|
|
|
|
|
|
file_or_url
|
2022-08-20 11:31:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2023-01-10 14:35:44 +01:00
|
|
|
fn op_require_resolve_exports<P>(
|
2022-08-20 11:31:33 -04:00
|
|
|
state: &mut OpState,
|
2022-12-19 17:09:54 +01:00
|
|
|
uses_local_node_modules_dir: bool,
|
2022-08-20 11:31:33 -04:00
|
|
|
modules_path: String,
|
|
|
|
_request: String,
|
|
|
|
name: String,
|
|
|
|
expansion: String,
|
|
|
|
parent_path: String,
|
2023-01-10 14:35:44 +01:00
|
|
|
) -> Result<Option<String>, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>().clone();
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
2022-08-20 11:31:33 -04:00
|
|
|
|
2022-12-19 17:09:54 +01:00
|
|
|
let pkg_path = if resolver.in_npm_package(&PathBuf::from(&modules_path))
|
|
|
|
&& !uses_local_node_modules_dir
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
modules_path
|
|
|
|
} else {
|
|
|
|
path_resolve(vec![modules_path, name])
|
|
|
|
};
|
|
|
|
let pkg = PackageJson::load(
|
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-20 11:31:33 -04:00
|
|
|
PathBuf::from(&pkg_path).join("package.json"),
|
|
|
|
)?;
|
|
|
|
|
|
|
|
if let Some(exports) = &pkg.exports {
|
2022-08-30 14:09:22 -04:00
|
|
|
let referrer = Url::from_file_path(parent_path).unwrap();
|
2022-08-20 11:31:33 -04:00
|
|
|
resolution::package_exports_resolve(
|
2022-08-30 14:09:22 -04:00
|
|
|
&pkg.path,
|
2023-01-27 10:43:16 -05:00
|
|
|
format!(".{expansion}"),
|
2022-08-20 11:31:33 -04:00
|
|
|
exports,
|
2022-08-30 14:09:22 -04:00
|
|
|
&referrer,
|
|
|
|
NodeModuleKind::Cjs,
|
2022-08-20 11:31:33 -04:00
|
|
|
resolution::REQUIRE_CONDITIONS,
|
2022-11-30 18:07:32 -05:00
|
|
|
NodeResolutionMode::Execution,
|
2022-08-20 11:31:33 -04:00
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-20 11:31:33 -04:00
|
|
|
)
|
2022-08-30 14:09:22 -04:00
|
|
|
.map(|r| Some(r.to_string_lossy().to_string()))
|
2022-08-20 11:31:33 -04:00
|
|
|
} else {
|
|
|
|
Ok(None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-30 14:09:22 -04:00
|
|
|
#[op]
|
|
|
|
fn op_require_read_closest_package_json<P>(
|
|
|
|
state: &mut OpState,
|
|
|
|
filename: String,
|
|
|
|
) -> Result<PackageJson, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
|
|
|
ensure_read_permission::<P>(
|
|
|
|
state,
|
|
|
|
PathBuf::from(&filename).parent().unwrap(),
|
|
|
|
)?;
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>().clone();
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
2022-08-30 14:09:22 -04:00
|
|
|
resolution::get_closest_package_json(
|
|
|
|
&Url::from_file_path(filename).unwrap(),
|
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-30 14:09:22 -04:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2022-08-20 11:31:33 -04:00
|
|
|
#[op]
|
2023-01-10 14:35:44 +01:00
|
|
|
fn op_require_read_package_scope<P>(
|
2022-08-20 11:31:33 -04:00
|
|
|
state: &mut OpState,
|
2022-09-12 15:47:54 -04:00
|
|
|
package_json_path: String,
|
2023-01-10 14:35:44 +01:00
|
|
|
) -> Option<PackageJson>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>().clone();
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
2022-09-12 15:47:54 -04:00
|
|
|
let package_json_path = PathBuf::from(package_json_path);
|
2023-01-10 14:35:44 +01:00
|
|
|
PackageJson::load(&*resolver, permissions, package_json_path).ok()
|
2022-08-20 11:31:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
#[op]
|
2022-08-24 18:07:49 +02:00
|
|
|
fn op_require_package_imports_resolve<P>(
|
2022-08-20 11:31:33 -04:00
|
|
|
state: &mut OpState,
|
|
|
|
parent_filename: String,
|
|
|
|
request: String,
|
2022-08-24 18:07:49 +02:00
|
|
|
) -> Result<Option<String>, AnyError>
|
|
|
|
where
|
|
|
|
P: NodePermissions + 'static,
|
|
|
|
{
|
2022-08-20 11:31:33 -04:00
|
|
|
let parent_path = PathBuf::from(&parent_filename);
|
2022-08-24 18:07:49 +02:00
|
|
|
ensure_read_permission::<P>(state, &parent_path)?;
|
2022-09-13 11:59:01 -04:00
|
|
|
let resolver = state.borrow::<Rc<dyn RequireNpmResolver>>().clone();
|
2023-01-10 14:35:44 +01:00
|
|
|
let permissions = state.borrow_mut::<P>();
|
|
|
|
let pkg = PackageJson::load(
|
|
|
|
&*resolver,
|
|
|
|
permissions,
|
|
|
|
parent_path.join("package.json"),
|
|
|
|
)?;
|
2022-08-20 11:31:33 -04:00
|
|
|
|
|
|
|
if pkg.imports.is_some() {
|
|
|
|
let referrer =
|
|
|
|
deno_core::url::Url::from_file_path(&parent_filename).unwrap();
|
|
|
|
let r = resolution::package_imports_resolve(
|
|
|
|
&request,
|
|
|
|
&referrer,
|
2022-08-30 14:09:22 -04:00
|
|
|
NodeModuleKind::Cjs,
|
2022-08-20 11:31:33 -04:00
|
|
|
resolution::REQUIRE_CONDITIONS,
|
2022-11-30 18:07:32 -05:00
|
|
|
NodeResolutionMode::Execution,
|
2022-08-20 11:31:33 -04:00
|
|
|
&*resolver,
|
2023-01-10 14:35:44 +01:00
|
|
|
permissions,
|
2022-08-20 11:31:33 -04:00
|
|
|
)
|
2022-08-30 14:09:22 -04:00
|
|
|
.map(|r| Some(Url::from_file_path(r).unwrap().to_string()));
|
2022-08-20 11:31:33 -04:00
|
|
|
state.put(resolver);
|
|
|
|
r
|
|
|
|
} else {
|
|
|
|
Ok(None)
|
|
|
|
}
|
2022-08-09 21:06:01 +02:00
|
|
|
}
|
2022-11-28 21:59:36 +01:00
|
|
|
|
|
|
|
#[op]
|
|
|
|
fn op_require_break_on_next_statement(state: &mut OpState) {
|
|
|
|
let inspector = state.borrow::<Rc<RefCell<JsRuntimeInspector>>>();
|
|
|
|
inspector
|
|
|
|
.borrow_mut()
|
|
|
|
.wait_for_session_and_break_on_next_statement()
|
|
|
|
}
|
2023-01-24 15:05:54 +01:00
|
|
|
|
|
|
|
pub struct NodeModulePolyfill {
|
|
|
|
/// Name of the module like "assert" or "timers/promises"
|
|
|
|
pub name: &'static str,
|
|
|
|
|
|
|
|
/// Specifier relative to the root of `deno_std` repo, like "node/assert.ts"
|
|
|
|
pub specifier: &'static str,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub static SUPPORTED_BUILTIN_NODE_MODULES: &[NodeModulePolyfill] = &[
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "assert",
|
|
|
|
specifier: "node/assert.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "assert/strict",
|
|
|
|
specifier: "node/assert/strict.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "async_hooks",
|
|
|
|
specifier: "node/async_hooks.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "buffer",
|
|
|
|
specifier: "node/buffer.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "child_process",
|
|
|
|
specifier: "node/child_process.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "cluster",
|
|
|
|
specifier: "node/cluster.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "console",
|
|
|
|
specifier: "node/console.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "constants",
|
|
|
|
specifier: "node/constants.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "crypto",
|
|
|
|
specifier: "node/crypto.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "dgram",
|
|
|
|
specifier: "node/dgram.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "dns",
|
|
|
|
specifier: "node/dns.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "dns/promises",
|
|
|
|
specifier: "node/dns/promises.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "domain",
|
|
|
|
specifier: "node/domain.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "events",
|
|
|
|
specifier: "node/events.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "fs",
|
|
|
|
specifier: "node/fs.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "fs/promises",
|
|
|
|
specifier: "node/fs/promises.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "http",
|
|
|
|
specifier: "node/http.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "https",
|
|
|
|
specifier: "node/https.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "module",
|
|
|
|
// NOTE(bartlomieju): `module` is special, because we don't want to use
|
|
|
|
// `deno_std/node/module.ts`, but instead use a special shim that we
|
|
|
|
// provide in `ext/node`.
|
|
|
|
specifier: "[USE `deno_node::MODULE_ES_SHIM` to get this module]",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "net",
|
|
|
|
specifier: "node/net.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "os",
|
|
|
|
specifier: "node/os.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "path",
|
|
|
|
specifier: "node/path.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "path/posix",
|
|
|
|
specifier: "node/path/posix.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "path/win32",
|
|
|
|
specifier: "node/path/win32.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "perf_hooks",
|
|
|
|
specifier: "node/perf_hooks.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "process",
|
|
|
|
specifier: "node/process.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "querystring",
|
|
|
|
specifier: "node/querystring.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "readline",
|
|
|
|
specifier: "node/readline.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "stream",
|
|
|
|
specifier: "node/stream.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "stream/consumers",
|
|
|
|
specifier: "node/stream/consumers.mjs",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "stream/promises",
|
|
|
|
specifier: "node/stream/promises.mjs",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "stream/web",
|
|
|
|
specifier: "node/stream/web.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "string_decoder",
|
|
|
|
specifier: "node/string_decoder.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "sys",
|
|
|
|
specifier: "node/sys.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "timers",
|
|
|
|
specifier: "node/timers.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "timers/promises",
|
|
|
|
specifier: "node/timers/promises.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "tls",
|
|
|
|
specifier: "node/tls.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "tty",
|
|
|
|
specifier: "node/tty.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "url",
|
|
|
|
specifier: "node/url.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "util",
|
|
|
|
specifier: "node/util.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "util/types",
|
|
|
|
specifier: "node/util/types.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "v8",
|
|
|
|
specifier: "node/v8.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "vm",
|
|
|
|
specifier: "node/vm.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "worker_threads",
|
|
|
|
specifier: "node/worker_threads.ts",
|
|
|
|
},
|
|
|
|
NodeModulePolyfill {
|
|
|
|
name: "zlib",
|
|
|
|
specifier: "node/zlib.ts",
|
|
|
|
},
|
|
|
|
];
|