// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license. use std::path::Path; use std::path::PathBuf; use std::time::Instant; use crate::Extension; use crate::InternalModuleLoaderCb; use crate::JsRuntime; use crate::RuntimeOptions; use crate::Snapshot; pub type CompressionCb = dyn Fn(&mut Vec, &[u8]); pub struct CreateSnapshotOptions { pub cargo_manifest_dir: &'static str, pub snapshot_path: PathBuf, pub startup_snapshot: Option, pub extensions: Vec, pub extensions_with_js: Vec, pub compression_cb: Option>, pub snapshot_module_load_cb: Option, } pub fn create_snapshot(create_snapshot_options: CreateSnapshotOptions) { let mut mark = Instant::now(); let js_runtime = JsRuntime::new(RuntimeOptions { will_snapshot: true, startup_snapshot: create_snapshot_options.startup_snapshot, extensions: create_snapshot_options.extensions, extensions_with_js: create_snapshot_options.extensions_with_js, snapshot_module_load_cb: create_snapshot_options.snapshot_module_load_cb, ..Default::default() }); let snapshot = js_runtime.snapshot(); let snapshot_slice: &[u8] = &snapshot; println!( "Snapshot size: {}, took {:#?} ({})", snapshot_slice.len(), Instant::now().saturating_duration_since(mark), create_snapshot_options.snapshot_path.display() ); mark = Instant::now(); let maybe_compressed_snapshot: Box> = if let Some(compression_cb) = create_snapshot_options.compression_cb { let mut vec = vec![]; vec.extend_from_slice( &u32::try_from(snapshot.len()) .expect("snapshot larger than 4gb") .to_le_bytes(), ); (compression_cb)(&mut vec, snapshot_slice); println!( "Snapshot compressed size: {}, took {:#?} ({})", vec.len(), Instant::now().saturating_duration_since(mark), create_snapshot_options.snapshot_path.display() ); mark = std::time::Instant::now(); Box::new(vec) } else { Box::new(snapshot_slice) }; std::fs::write( &create_snapshot_options.snapshot_path, &*maybe_compressed_snapshot, ) .unwrap(); println!( "Snapshot written, took: {:#?} ({})", Instant::now().saturating_duration_since(mark), create_snapshot_options.snapshot_path.display(), ); } pub type FilterFn = Box bool>; pub fn get_js_files( cargo_manifest_dir: &'static str, directory: &str, filter: Option, ) -> Vec { let manifest_dir = Path::new(cargo_manifest_dir); let mut js_files = std::fs::read_dir(directory) .unwrap() .map(|dir_entry| { let file = dir_entry.unwrap(); manifest_dir.join(file.path()) }) .filter(|path| { path.extension().unwrap_or_default() == "js" && filter.as_ref().map(|filter| filter(path)).unwrap_or(true) }) .collect::>(); js_files.sort(); js_files }