2023-01-13 15:51:32 +08:00
|
|
|
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
|
|
|
2023-02-07 20:22:46 +01:00
|
|
|
use anyhow::Context;
|
2023-01-13 15:51:32 +08:00
|
|
|
use std::path::Path;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
|
2022-11-21 14:36:26 +01:00
|
|
|
use crate::Extension;
|
|
|
|
use crate::JsRuntime;
|
2023-02-07 20:22:46 +01:00
|
|
|
use crate::ModuleSpecifier;
|
2022-11-21 14:36:26 +01:00
|
|
|
use crate::RuntimeOptions;
|
|
|
|
use crate::Snapshot;
|
|
|
|
|
|
|
|
pub type CompressionCb = dyn Fn(&mut Vec<u8>, &[u8]);
|
|
|
|
|
|
|
|
pub struct CreateSnapshotOptions {
|
|
|
|
pub cargo_manifest_dir: &'static str,
|
|
|
|
pub snapshot_path: PathBuf,
|
|
|
|
pub startup_snapshot: Option<Snapshot>,
|
|
|
|
pub extensions: Vec<Extension>,
|
2022-11-27 00:58:23 +01:00
|
|
|
pub extensions_with_js: Vec<Extension>,
|
2022-11-21 14:36:26 +01:00
|
|
|
pub additional_files: Vec<PathBuf>,
|
2023-02-07 20:22:46 +01:00
|
|
|
pub additional_esm_files: Vec<PathBuf>,
|
2022-11-21 14:36:26 +01:00
|
|
|
pub compression_cb: Option<Box<CompressionCb>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn create_snapshot(create_snapshot_options: CreateSnapshotOptions) {
|
|
|
|
let mut js_runtime = JsRuntime::new(RuntimeOptions {
|
|
|
|
will_snapshot: true,
|
|
|
|
startup_snapshot: create_snapshot_options.startup_snapshot,
|
|
|
|
extensions: create_snapshot_options.extensions,
|
2022-11-27 00:58:23 +01:00
|
|
|
extensions_with_js: create_snapshot_options.extensions_with_js,
|
2022-11-21 14:36:26 +01:00
|
|
|
..Default::default()
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO(nayeemrmn): https://github.com/rust-lang/cargo/issues/3946 to get the
|
|
|
|
// workspace root.
|
|
|
|
let display_root = Path::new(create_snapshot_options.cargo_manifest_dir)
|
|
|
|
.parent()
|
|
|
|
.unwrap();
|
|
|
|
for file in create_snapshot_options.additional_files {
|
|
|
|
let display_path = file.strip_prefix(display_root).unwrap_or(&file);
|
|
|
|
let display_path_str = display_path.display().to_string();
|
|
|
|
js_runtime
|
|
|
|
.execute_script(
|
2023-02-05 17:49:20 +01:00
|
|
|
&("internal:".to_string() + &display_path_str.replace('\\', "/")),
|
2022-11-21 14:36:26 +01:00
|
|
|
&std::fs::read_to_string(&file).unwrap(),
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
}
|
2023-02-07 20:22:46 +01:00
|
|
|
for file in create_snapshot_options.additional_esm_files {
|
|
|
|
let display_path = file.strip_prefix(display_root).unwrap_or(&file);
|
|
|
|
let display_path_str = display_path.display().to_string();
|
|
|
|
|
|
|
|
let filename =
|
|
|
|
&("internal:".to_string() + &display_path_str.replace('\\', "/"));
|
|
|
|
|
|
|
|
futures::executor::block_on(async {
|
|
|
|
let id = js_runtime
|
|
|
|
.load_side_module(
|
|
|
|
&ModuleSpecifier::parse(filename)?,
|
|
|
|
Some(std::fs::read_to_string(&file)?),
|
|
|
|
)
|
|
|
|
.await?;
|
|
|
|
let receiver = js_runtime.mod_evaluate(id);
|
|
|
|
js_runtime.run_event_loop(false).await?;
|
|
|
|
receiver.await?
|
|
|
|
})
|
|
|
|
.with_context(|| format!("Couldn't execute '{}'", file.display()))
|
|
|
|
.unwrap();
|
|
|
|
}
|
2022-11-21 14:36:26 +01:00
|
|
|
|
|
|
|
let snapshot = js_runtime.snapshot();
|
|
|
|
let snapshot_slice: &[u8] = &snapshot;
|
|
|
|
println!("Snapshot size: {}", snapshot_slice.len());
|
|
|
|
|
|
|
|
let maybe_compressed_snapshot: Box<dyn AsRef<[u8]>> =
|
|
|
|
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: {}", vec.len());
|
|
|
|
|
|
|
|
Box::new(vec)
|
|
|
|
} else {
|
|
|
|
Box::new(snapshot_slice)
|
|
|
|
};
|
|
|
|
|
|
|
|
std::fs::write(
|
|
|
|
&create_snapshot_options.snapshot_path,
|
|
|
|
&*maybe_compressed_snapshot,
|
|
|
|
)
|
|
|
|
.unwrap();
|
|
|
|
println!(
|
|
|
|
"Snapshot written to: {} ",
|
|
|
|
create_snapshot_options.snapshot_path.display()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-02-07 20:22:46 +01:00
|
|
|
pub type FilterFn = Box<dyn Fn(&PathBuf) -> bool>;
|
|
|
|
|
2022-11-21 14:36:26 +01:00
|
|
|
pub fn get_js_files(
|
|
|
|
cargo_manifest_dir: &'static str,
|
|
|
|
directory: &str,
|
2023-02-07 20:22:46 +01:00
|
|
|
filter: Option<FilterFn>,
|
2022-11-21 14:36:26 +01:00
|
|
|
) -> Vec<PathBuf> {
|
|
|
|
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"
|
2023-02-07 20:22:46 +01:00
|
|
|
&& filter.as_ref().map(|filter| filter(path)).unwrap_or(true)
|
2022-11-21 14:36:26 +01:00
|
|
|
})
|
|
|
|
.collect::<Vec<PathBuf>>();
|
|
|
|
js_files.sort();
|
|
|
|
js_files
|
|
|
|
}
|