summaryrefslogtreecommitdiff
path: root/cli
diff options
context:
space:
mode:
Diffstat (limited to 'cli')
-rw-r--r--cli/Cargo.toml7
-rw-r--r--cli/cache.rs16
-rw-r--r--cli/main.rs84
-rw-r--r--cli/standalone.rs122
-rw-r--r--cli/tests/integration/compile_tests.rs35
-rw-r--r--cli/tests/testdata/standalone_dynamic_imports.ts15
-rw-r--r--cli/tests/testdata/standalone_error_module_with_imports_1.ts1
-rw-r--r--cli/tests/testdata/standalone_error_module_with_imports_2.ts2
-rw-r--r--cli/tools/doc.rs2
-rw-r--r--cli/tools/standalone.rs21
-rw-r--r--cli/tsc.rs2
11 files changed, 174 insertions, 133 deletions
diff --git a/cli/Cargo.toml b/cli/Cargo.toml
index 1943a635f..702b2cf0f 100644
--- a/cli/Cargo.toml
+++ b/cli/Cargo.toml
@@ -47,8 +47,8 @@ winres = "=0.1.11"
[dependencies]
deno_ast = { version = "0.11.0", features = ["bundler", "codegen", "dep_graph", "module_specifier", "proposal", "react", "sourcemap", "transforms", "transpiling", "typescript", "view", "visit"] }
deno_core = { version = "0.118.0", path = "../core" }
-deno_doc = "0.29.0"
-deno_graph = "0.22.0"
+deno_doc = "0.31.0"
+deno_graph = "0.23.0"
deno_lint = { version = "0.24.0", features = ["docs"] }
deno_runtime = { version = "0.44.0", path = "../runtime" }
@@ -63,9 +63,10 @@ data-url = "=0.1.1"
dissimilar = "=1.0.2"
dprint-plugin-json = "=0.14.1"
dprint-plugin-markdown = "=0.12.2"
-dprint-plugin-typescript = "=0.62.2"
+dprint-plugin-typescript = "=0.64.1"
encoding_rs = "=0.8.29"
env_logger = "=0.8.4"
+eszip = "=0.16.0"
fancy-regex = "=0.7.1"
http = "=0.2.4"
import_map = "=0.8.0"
diff --git a/cli/cache.rs b/cli/cache.rs
index 1075230d1..8bd40fc8a 100644
--- a/cli/cache.rs
+++ b/cli/cache.rs
@@ -164,7 +164,7 @@ impl Loader for FetchCacher {
Err(err)
},
|file| {
- Ok(Some(LoadResponse {
+ Ok(Some(LoadResponse::Module {
specifier: file.specifier,
maybe_headers: file.maybe_headers,
content: file.source,
@@ -288,11 +288,15 @@ impl Loader for MemoryCacher {
specifier_str = specifier_str[3..].to_string();
}
}
- let response = self.sources.get(&specifier_str).map(|c| LoadResponse {
- specifier: specifier.clone(),
- maybe_headers: None,
- content: c.to_owned(),
- });
+ let response =
+ self
+ .sources
+ .get(&specifier_str)
+ .map(|c| LoadResponse::Module {
+ specifier: specifier.clone(),
+ maybe_headers: None,
+ content: c.to_owned(),
+ });
Box::pin(future::ready(Ok(response)))
}
}
diff --git a/cli/main.rs b/cli/main.rs
index f8c8e695a..ca6b36f0a 100644
--- a/cli/main.rs
+++ b/cli/main.rs
@@ -415,23 +415,14 @@ async fn compile_command(
"An executable name was not provided. One could not be inferred from the URL. Aborting.",
))?;
- let graph =
- create_graph_and_maybe_check(module_specifier.clone(), &ps, debug).await?;
-
- let source = (graph.as_ref().modules().len() == 1)
- .then(|| {
- let root_module = graph.as_ref().modules()[0];
- match root_module.media_type {
- MediaType::JavaScript if root_module.maybe_source.is_some() => {
- Some(Ok(root_module.maybe_source.clone().unwrap().to_string()))
- }
- _ => None,
- }
- })
- .flatten()
- .unwrap_or_else(|| {
- bundle_module_graph(graph.as_ref(), &ps, &ps.flags).map(|r| r.0)
- })?;
+ let graph = Arc::try_unwrap(
+ create_graph_and_maybe_check(module_specifier.clone(), &ps, debug).await?,
+ )
+ .map_err(|_| {
+ generic_error("There should only be one reference to ModuleGraph")
+ })?;
+
+ let eszip = eszip::EszipV2::from_graph(graph, Default::default())?;
info!(
"{} {}",
@@ -446,7 +437,8 @@ async fn compile_command(
let final_bin = tools::standalone::create_standalone_binary(
original_binary,
- source,
+ eszip,
+ module_specifier.clone(),
run_flags,
)?;
@@ -1430,34 +1422,40 @@ pub fn main() {
colors::enable_ansi(); // For Windows 10
let args: Vec<String> = env::args().collect();
- let standalone_res = match standalone::extract_standalone(args.clone()) {
- Ok(Some((metadata, bundle))) => {
- run_basic(standalone::run(bundle, metadata))
- }
- Ok(None) => Ok(()),
- Err(err) => Err(err),
- };
- // TODO(bartlomieju): doesn't handle exit code set by the runtime properly
- unwrap_or_exit(standalone_res);
-
- let flags = match flags::flags_from_vec(args) {
- Ok(flags) => flags,
- Err(err @ clap::Error { .. })
- if err.kind == clap::ErrorKind::DisplayHelp
- || err.kind == clap::ErrorKind::DisplayVersion =>
- {
- err.print().unwrap();
- std::process::exit(0);
+
+ let exit_code = async move {
+ let standalone_res =
+ match standalone::extract_standalone(args.clone()).await {
+ Ok(Some((metadata, eszip))) => standalone::run(eszip, metadata).await,
+ Ok(None) => Ok(()),
+ Err(err) => Err(err),
+ };
+ // TODO(bartlomieju): doesn't handle exit code set by the runtime properly
+ unwrap_or_exit(standalone_res);
+
+ let flags = match flags::flags_from_vec(args) {
+ Ok(flags) => flags,
+ Err(err @ clap::Error { .. })
+ if err.kind == clap::ErrorKind::DisplayHelp
+ || err.kind == clap::ErrorKind::DisplayVersion =>
+ {
+ err.print().unwrap();
+ std::process::exit(0);
+ }
+ Err(err) => unwrap_or_exit(Err(AnyError::from(err))),
+ };
+ if !flags.v8_flags.is_empty() {
+ init_v8_flags(&*flags.v8_flags);
}
- Err(err) => unwrap_or_exit(Err(AnyError::from(err))),
- };
- if !flags.v8_flags.is_empty() {
- init_v8_flags(&*flags.v8_flags);
- }
- logger::init(flags.log_level);
+ logger::init(flags.log_level);
+
+ let exit_code = get_subcommand(flags).await;
+
+ exit_code
+ };
- let exit_code = unwrap_or_exit(run_basic(get_subcommand(flags)));
+ let exit_code = unwrap_or_exit(run_basic(exit_code));
std::process::exit(exit_code);
}
diff --git a/cli/standalone.rs b/cli/standalone.rs
index c47ec5736..9fa210795 100644
--- a/cli/standalone.rs
+++ b/cli/standalone.rs
@@ -12,7 +12,6 @@ use deno_core::error::type_error;
use deno_core::error::AnyError;
use deno_core::futures::FutureExt;
use deno_core::located_script_name;
-use deno_core::resolve_url;
use deno_core::serde::Deserialize;
use deno_core::serde::Serialize;
use deno_core::serde_json;
@@ -31,16 +30,14 @@ use deno_runtime::worker::WorkerOptions;
use deno_runtime::BootstrapOptions;
use log::Level;
use std::env::current_exe;
-use std::fs::File;
use std::io::BufReader;
use std::io::Cursor;
-use std::io::Read;
-use std::io::Seek;
use std::io::SeekFrom;
use std::iter::once;
use std::pin::Pin;
use std::rc::Rc;
use std::sync::Arc;
+use tokio::io::{AsyncReadExt, AsyncSeekExt};
#[derive(Deserialize, Serialize)]
pub struct Metadata {
@@ -54,6 +51,7 @@ pub struct Metadata {
pub ca_stores: Option<Vec<String>>,
pub ca_data: Option<Vec<u8>>,
pub unsafely_ignore_certificate_errors: Option<Vec<String>>,
+ pub entrypoint: ModuleSpecifier,
}
pub const MAGIC_TRAILER: &[u8; 8] = b"d3n0l4nd";
@@ -67,37 +65,51 @@ pub const MAGIC_TRAILER: &[u8; 8] = b"d3n0l4nd";
/// These are dereferenced, and the bundle is executed under the configuration
/// specified by the metadata. If no magic trailer is present, this function
/// exits with `Ok(None)`.
-pub fn extract_standalone(
+pub async fn extract_standalone(
args: Vec<String>,
-) -> Result<Option<(Metadata, String)>, AnyError> {
+) -> Result<Option<(Metadata, eszip::EszipV2)>, AnyError> {
let current_exe_path = current_exe()?;
- let mut current_exe = File::open(current_exe_path)?;
- let trailer_pos = current_exe.seek(SeekFrom::End(-24))?;
+ let file = tokio::fs::File::open(current_exe_path).await?;
+
+ let mut bufreader = tokio::io::BufReader::new(file);
+
+ let trailer_pos = bufreader.seek(SeekFrom::End(-24)).await?;
let mut trailer = [0; 24];
- current_exe.read_exact(&mut trailer)?;
+ bufreader.read_exact(&mut trailer).await?;
let (magic_trailer, rest) = trailer.split_at(8);
if magic_trailer != MAGIC_TRAILER {
return Ok(None);
}
- let (bundle_pos, rest) = rest.split_at(8);
+ let (eszip_archive_pos, rest) = rest.split_at(8);
let metadata_pos = rest;
- let bundle_pos = u64_from_bytes(bundle_pos)?;
+ let eszip_archive_pos = u64_from_bytes(eszip_archive_pos)?;
let metadata_pos = u64_from_bytes(metadata_pos)?;
- let bundle_len = metadata_pos - bundle_pos;
let metadata_len = trailer_pos - metadata_pos;
- current_exe.seek(SeekFrom::Start(bundle_pos))?;
- let bundle = read_string_slice(&mut current_exe, bundle_pos, bundle_len)
- .context("Failed to read source bundle from the current executable")?;
- let metadata =
- read_string_slice(&mut current_exe, metadata_pos, metadata_len)
- .context("Failed to read metadata from the current executable")?;
+ bufreader.seek(SeekFrom::Start(eszip_archive_pos)).await?;
+
+ let (eszip, loader) = eszip::EszipV2::parse(bufreader)
+ .await
+ .context("Failed to parse eszip header")?;
+
+ let mut bufreader = loader.await.context("Failed to parse eszip archive")?;
+
+ bufreader.seek(SeekFrom::Start(metadata_pos)).await?;
+
+ let mut metadata = String::new();
+
+ bufreader
+ .take(metadata_len)
+ .read_to_string(&mut metadata)
+ .await
+ .context("Failed to read metadata from the current executable")?;
let mut metadata: Metadata = serde_json::from_str(&metadata).unwrap();
metadata.argv.append(&mut args[1..].to_vec());
- Ok(Some((metadata, bundle)))
+
+ Ok(Some((metadata, eszip)))
}
fn u64_from_bytes(arr: &[u8]) -> Result<u64, AnyError> {
@@ -107,39 +119,17 @@ fn u64_from_bytes(arr: &[u8]) -> Result<u64, AnyError> {
Ok(u64::from_be_bytes(*fixed_arr))
}
-fn read_string_slice(
- file: &mut File,
- pos: u64,
- len: u64,
-) -> Result<String, AnyError> {
- let mut string = String::new();
- file.seek(SeekFrom::Start(pos))?;
- file.take(len).read_to_string(&mut string)?;
- // TODO: check amount of bytes read
- Ok(string)
-}
-
-const SPECIFIER: &str = "file://$deno$/bundle.js";
-
-struct EmbeddedModuleLoader(String);
+struct EmbeddedModuleLoader(eszip::EszipV2);
impl ModuleLoader for EmbeddedModuleLoader {
fn resolve(
&self,
specifier: &str,
- _referrer: &str,
+ base: &str,
_is_main: bool,
) -> Result<ModuleSpecifier, AnyError> {
- if let Ok(module_specifier) = resolve_url(specifier) {
- if get_source_from_data_url(&module_specifier).is_ok()
- || specifier == SPECIFIER
- {
- return Ok(module_specifier);
- }
- }
- Err(type_error(
- "Self-contained binaries don't support module loading",
- ))
+ let resolve = deno_core::resolve_import(specifier, base)?;
+ Ok(resolve)
}
fn load(
@@ -149,22 +139,36 @@ impl ModuleLoader for EmbeddedModuleLoader {
_is_dynamic: bool,
) -> Pin<Box<deno_core::ModuleSourceFuture>> {
let module_specifier = module_specifier.clone();
+
let is_data_uri = get_source_from_data_url(&module_specifier).ok();
- let code = if let Some((ref source, _)) = is_data_uri {
- source.to_string()
- } else {
- self.0.to_string()
- };
+
+ let module = self
+ .0
+ .get_module(module_specifier.as_str())
+ .ok_or_else(|| type_error("Module not found"));
+
async move {
- if is_data_uri.is_none() && module_specifier.to_string() != SPECIFIER {
- return Err(type_error(
- "Self-contained binaries don't support module loading",
- ));
+ if let Some((ref source, _)) = is_data_uri {
+ return Ok(deno_core::ModuleSource {
+ code: source.to_owned(),
+ module_type: deno_core::ModuleType::JavaScript,
+ module_url_specified: module_specifier.to_string(),
+ module_url_found: module_specifier.to_string(),
+ });
}
+ let module = module?;
+ let code = module.source().await;
+ let code = std::str::from_utf8(&code)
+ .map_err(|_| type_error("Module source is not utf-8"))?
+ .to_owned();
+
Ok(deno_core::ModuleSource {
code,
- module_type: deno_core::ModuleType::JavaScript,
+ module_type: match module.kind {
+ eszip::ModuleKind::JavaScript => deno_core::ModuleType::JavaScript,
+ eszip::ModuleKind::Json => deno_core::ModuleType::Json,
+ },
module_url_specified: module_specifier.to_string(),
module_url_found: module_specifier.to_string(),
})
@@ -195,16 +199,16 @@ fn metadata_to_flags(metadata: &Metadata) -> Flags {
}
pub async fn run(
- source_code: String,
+ eszip: eszip::EszipV2,
metadata: Metadata,
) -> Result<(), AnyError> {
let flags = metadata_to_flags(&metadata);
- let main_module = resolve_url(SPECIFIER)?;
+ let main_module = &metadata.entrypoint;
let ps = ProcState::build(Arc::new(flags)).await?;
let permissions = Permissions::from_options(&metadata.permissions);
let blob_store = BlobStore::default();
let broadcast_channel = InMemoryBroadcastChannel::default();
- let module_loader = Rc::new(EmbeddedModuleLoader(source_code));
+ let module_loader = Rc::new(EmbeddedModuleLoader(eszip));
let create_web_worker_cb = Arc::new(|_| {
todo!("Worker are currently not supported in standalone binaries");
});
@@ -276,7 +280,7 @@ pub async fn run(
permissions,
options,
);
- worker.execute_main_module(&main_module).await?;
+ worker.execute_main_module(main_module).await?;
worker.dispatch_load_event(&located_script_name!())?;
worker.run_event_loop(true).await?;
worker.dispatch_unload_event(&located_script_name!())?;
diff --git a/cli/tests/integration/compile_tests.rs b/cli/tests/integration/compile_tests.rs
index 1a81784af..dcbbf158b 100644
--- a/cli/tests/integration/compile_tests.rs
+++ b/cli/tests/integration/compile_tests.rs
@@ -107,8 +107,9 @@ fn standalone_error() {
} else {
dir.path().join("error")
};
+ let testdata_path = util::testdata_path();
let output = util::deno_cmd()
- .current_dir(util::testdata_path())
+ .current_dir(&testdata_path)
.arg("compile")
.arg("--unstable")
.arg("--output")
@@ -130,26 +131,32 @@ fn standalone_error() {
.unwrap();
assert!(!output.status.success());
assert_eq!(output.stdout, b"");
- let expected_stderr = "error: Error: boom!\n at boom (file://$deno$/bundle.js:6:11)\n at foo (file://$deno$/bundle.js:9:5)\n at file://$deno$/bundle.js:11:1\n";
let stderr = String::from_utf8(output.stderr).unwrap();
- assert_eq!(stderr, expected_stderr);
+ // On Windows, we cannot assert the file path (because '\').
+ // Instead we just check for relevant output.
+ assert!(stderr.contains("error: Error: boom!\n at boom (file://"));
+ assert!(stderr.contains("standalone_error.ts:2:11"));
+ assert!(stderr.contains("at foo (file://"));
+ assert!(stderr.contains("standalone_error.ts:5:5"));
+ assert!(stderr.contains("standalone_error.ts:7:1"));
}
#[test]
-fn standalone_no_module_load() {
+fn standalone_error_module_with_imports() {
let dir = TempDir::new().expect("tempdir fail");
let exe = if cfg!(windows) {
- dir.path().join("hello.exe")
+ dir.path().join("error.exe")
} else {
- dir.path().join("hello")
+ dir.path().join("error")
};
+ let testdata_path = util::testdata_path();
let output = util::deno_cmd()
- .current_dir(util::testdata_path())
+ .current_dir(&testdata_path)
.arg("compile")
.arg("--unstable")
.arg("--output")
.arg(&exe)
- .arg("./standalone_import.ts")
+ .arg("./standalone_error_module_with_imports_1.ts")
.stdout(std::process::Stdio::piped())
.spawn()
.unwrap()
@@ -157,6 +164,7 @@ fn standalone_no_module_load() {
.unwrap();
assert!(output.status.success());
let output = Command::new(exe)
+ .env("NO_COLOR", "1")
.stdout(std::process::Stdio::piped())
.stderr(std::process::Stdio::piped())
.spawn()
@@ -164,10 +172,13 @@ fn standalone_no_module_load() {
.wait_with_output()
.unwrap();
assert!(!output.status.success());
- assert_eq!(output.stdout, b"start\n");
- let stderr_str = String::from_utf8(output.stderr).unwrap();
- assert!(util::strip_ansi_codes(&stderr_str)
- .contains("Self-contained binaries don't support module loading"));
+ println!("{:#?}", &output);
+ assert_eq!(output.stdout, b"hello\n");
+ let stderr = String::from_utf8(output.stderr).unwrap();
+ // On Windows, we cannot assert the file path (because '\').
+ // Instead we just check for relevant output.
+ assert!(stderr.contains("error: Error: boom!\n at file://"));
+ assert!(stderr.contains("standalone_error_module_with_imports_2.ts:2:7"));
}
#[test]
diff --git a/cli/tests/testdata/standalone_dynamic_imports.ts b/cli/tests/testdata/standalone_dynamic_imports.ts
new file mode 100644
index 000000000..d4dc55152
--- /dev/null
+++ b/cli/tests/testdata/standalone_dynamic_imports.ts
@@ -0,0 +1,15 @@
+(async () => {
+ const { returnsHi, returnsFoo2, printHello3 } = await import(
+ "./subdir/mod1.ts"
+ );
+
+ printHello3();
+
+ if (returnsHi() !== "Hi") {
+ throw Error("Unexpected");
+ }
+
+ if (returnsFoo2() !== "Foo") {
+ throw Error("Unexpected");
+ }
+})();
diff --git a/cli/tests/testdata/standalone_error_module_with_imports_1.ts b/cli/tests/testdata/standalone_error_module_with_imports_1.ts
new file mode 100644
index 000000000..bf38f7263
--- /dev/null
+++ b/cli/tests/testdata/standalone_error_module_with_imports_1.ts
@@ -0,0 +1 @@
+import "./standalone_error_module_with_imports_2.ts";
diff --git a/cli/tests/testdata/standalone_error_module_with_imports_2.ts b/cli/tests/testdata/standalone_error_module_with_imports_2.ts
new file mode 100644
index 000000000..ef052b512
--- /dev/null
+++ b/cli/tests/testdata/standalone_error_module_with_imports_2.ts
@@ -0,0 +1,2 @@
+console.log("hello");
+throw new Error("boom!");
diff --git a/cli/tools/doc.rs b/cli/tools/doc.rs
index 38f8c2918..36fc34e5b 100644
--- a/cli/tools/doc.rs
+++ b/cli/tools/doc.rs
@@ -81,7 +81,7 @@ impl Loader for DocLoader {
.fetch(&specifier, &mut Permissions::allow_all())
.await
.map(|file| {
- Some(LoadResponse {
+ Some(LoadResponse::Module {
specifier,
content: file.source.clone(),
maybe_headers: file.maybe_headers,
diff --git a/cli/tools/standalone.rs b/cli/tools/standalone.rs
index 2f8ca1533..44150add8 100644
--- a/cli/tools/standalone.rs
+++ b/cli/tools/standalone.rs
@@ -8,6 +8,7 @@ use crate::flags::RunFlags;
use deno_core::anyhow::bail;
use deno_core::error::AnyError;
use deno_core::serde_json;
+use deno_graph::ModuleSpecifier;
use deno_runtime::deno_fetch::reqwest::Client;
use std::env;
use std::fs::read;
@@ -86,10 +87,12 @@ async fn download_base_binary(
/// and magic trailer to the currently executing binary.
pub fn create_standalone_binary(
mut original_bin: Vec<u8>,
- source_code: String,
+ eszip: eszip::EszipV2,
+ entrypoint: ModuleSpecifier,
flags: Flags,
) -> Result<Vec<u8>, AnyError> {
- let mut source_code = source_code.as_bytes().to_vec();
+ let mut eszip_archive = eszip.into_bytes();
+
let ca_data = match &flags.ca_file {
Some(ca_file) => Some(read(ca_file)?),
None => None,
@@ -107,19 +110,21 @@ pub fn create_standalone_binary(
log_level: flags.log_level,
ca_stores: flags.ca_stores,
ca_data,
+ entrypoint,
};
let mut metadata = serde_json::to_string(&metadata)?.as_bytes().to_vec();
- let bundle_pos = original_bin.len();
- let metadata_pos = bundle_pos + source_code.len();
+ let eszip_pos = original_bin.len();
+ let metadata_pos = eszip_pos + eszip_archive.len();
let mut trailer = MAGIC_TRAILER.to_vec();
- trailer.write_all(&bundle_pos.to_be_bytes())?;
+ trailer.write_all(&eszip_pos.to_be_bytes())?;
trailer.write_all(&metadata_pos.to_be_bytes())?;
- let mut final_bin =
- Vec::with_capacity(original_bin.len() + source_code.len() + trailer.len());
+ let mut final_bin = Vec::with_capacity(
+ original_bin.len() + eszip_archive.len() + trailer.len(),
+ );
final_bin.append(&mut original_bin);
- final_bin.append(&mut source_code);
+ final_bin.append(&mut eszip_archive);
final_bin.append(&mut metadata);
final_bin.append(&mut trailer);
diff --git a/cli/tsc.rs b/cli/tsc.rs
index 0e55d27e2..b679b049e 100644
--- a/cli/tsc.rs
+++ b/cli/tsc.rs
@@ -741,7 +741,7 @@ mod tests {
let source_path = self.fixtures.join(specifier_text);
let response = fs::read_to_string(&source_path)
.map(|c| {
- Some(deno_graph::source::LoadResponse {
+ Some(deno_graph::source::LoadResponse::Module {
specifier: specifier.clone(),
maybe_headers: None,
content: Arc::new(c),