diff options
Diffstat (limited to 'cli/standalone.rs')
-rw-r--r-- | cli/standalone.rs | 159 |
1 files changed, 159 insertions, 0 deletions
diff --git a/cli/standalone.rs b/cli/standalone.rs new file mode 100644 index 000000000..805849c81 --- /dev/null +++ b/cli/standalone.rs @@ -0,0 +1,159 @@ +use crate::colors; +use crate::flags::Flags; +use crate::permissions::Permissions; +use crate::program_state::ProgramState; +use crate::tokio_util; +use crate::worker::MainWorker; +use deno_core::error::type_error; +use deno_core::error::AnyError; +use deno_core::futures::FutureExt; +use deno_core::ModuleLoader; +use deno_core::ModuleSpecifier; +use deno_core::OpState; +use std::cell::RefCell; +use std::convert::TryInto; +use std::env::current_exe; +use std::fs::File; +use std::io::Read; +use std::io::Seek; +use std::io::SeekFrom; +use std::io::Write; +use std::pin::Pin; +use std::rc::Rc; + +const MAGIC_TRAILER: &[u8; 8] = b"d3n0l4nd"; + +/// This function will try to run this binary as a standalone binary +/// produced by `deno compile`. It determines if this is a stanalone +/// binary by checking for the magic trailer string `D3N0` at EOF-12. +/// After the magic trailer is a u64 pointer to the start of the JS +/// file embedded in the binary. This file is read, and run. If no +/// magic trailer is present, this function exits with Ok(()). +pub fn try_run_standalone_binary(args: Vec<String>) -> Result<(), AnyError> { + let current_exe_path = current_exe()?; + + let mut current_exe = File::open(current_exe_path)?; + let trailer_pos = current_exe.seek(SeekFrom::End(-16))?; + let mut trailer = [0; 16]; + current_exe.read_exact(&mut trailer)?; + let (magic_trailer, bundle_pos_arr) = trailer.split_at(8); + if magic_trailer == MAGIC_TRAILER { + let bundle_pos_arr: &[u8; 8] = bundle_pos_arr.try_into()?; + let bundle_pos = u64::from_be_bytes(*bundle_pos_arr); + current_exe.seek(SeekFrom::Start(bundle_pos))?; + + let bundle_len = trailer_pos - bundle_pos; + let mut bundle = String::new(); + current_exe.take(bundle_len).read_to_string(&mut bundle)?; + // TODO: check amount of bytes read + + if let Err(err) = tokio_util::run_basic(run(bundle, args)) { + eprintln!("{}: {}", colors::red_bold("error"), err.to_string()); + std::process::exit(1); + } + std::process::exit(0); + } else { + Ok(()) + } +} + +const SPECIFIER: &str = "file://$deno$/bundle.js"; + +struct EmbeddedModuleLoader(String); + +impl ModuleLoader for EmbeddedModuleLoader { + fn resolve( + &self, + _op_state: Rc<RefCell<OpState>>, + specifier: &str, + _referrer: &str, + _is_main: bool, + ) -> Result<ModuleSpecifier, AnyError> { + if specifier != SPECIFIER { + return Err(type_error( + "Self-contained binaries don't support module loading", + )); + } + Ok(ModuleSpecifier::resolve_url(specifier)?) + } + + fn load( + &self, + _op_state: Rc<RefCell<OpState>>, + module_specifier: &ModuleSpecifier, + _maybe_referrer: Option<ModuleSpecifier>, + _is_dynamic: bool, + ) -> Pin<Box<deno_core::ModuleSourceFuture>> { + let module_specifier = module_specifier.clone(); + let code = self.0.to_string(); + async move { + if module_specifier.to_string() != SPECIFIER { + return Err(type_error( + "Self-contained binaries don't support module loading", + )); + } + Ok(deno_core::ModuleSource { + code, + module_url_specified: module_specifier.to_string(), + module_url_found: module_specifier.to_string(), + }) + } + .boxed_local() + } +} + +async fn run(source_code: String, args: Vec<String>) -> Result<(), AnyError> { + let mut flags = Flags::default(); + flags.argv = args[1..].to_vec(); + // TODO(lucacasonato): remove once you can specify this correctly through embedded metadata + flags.unstable = true; + let main_module = ModuleSpecifier::resolve_url(SPECIFIER)?; + let program_state = ProgramState::new(flags.clone())?; + let permissions = Permissions::allow_all(); + let module_loader = Rc::new(EmbeddedModuleLoader(source_code)); + let mut worker = MainWorker::from_options( + &program_state, + main_module.clone(), + permissions, + module_loader, + ); + worker.execute_module(&main_module).await?; + worker.execute("window.dispatchEvent(new Event('load'))")?; + worker.run_event_loop().await?; + worker.execute("window.dispatchEvent(new Event('unload'))")?; + Ok(()) +} + +/// This functions creates a standalone deno binary by appending a bundle +/// and magic trailer to the currently executing binary. +pub async fn create_standalone_binary( + mut source_code: Vec<u8>, + out_file: String, +) -> Result<(), AnyError> { + let original_binary_path = std::env::current_exe()?; + let mut original_bin = tokio::fs::read(original_binary_path).await?; + + let mut trailer = MAGIC_TRAILER.to_vec(); + trailer.write_all(&original_bin.len().to_be_bytes())?; + + let mut final_bin = + Vec::with_capacity(original_bin.len() + source_code.len() + trailer.len()); + final_bin.append(&mut original_bin); + final_bin.append(&mut source_code); + final_bin.append(&mut trailer); + + let out_file = if cfg!(windows) && !out_file.ends_with(".exe") { + format!("{}.exe", out_file) + } else { + out_file + }; + tokio::fs::write(&out_file, final_bin).await?; + #[cfg(unix)] + { + use std::os::unix::fs::PermissionsExt; + let perms = std::fs::Permissions::from_mode(0o777); + tokio::fs::set_permissions(out_file, perms).await?; + } + + Ok(()) +} |