Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(compat): CJS/ESM interoperability #13553

Merged
merged 38 commits into from
Feb 27, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
ac6f556
compat: CJS/ESM interoperability
bartlomieju Jan 31, 2022
3ec042c
example translation
bartlomieju Jan 31, 2022
91a7af7
example translation with reexports
bartlomieju Jan 31, 2022
8f42232
working prototype
bartlomieju Feb 1, 2022
7ce82b2
try to handle reexports
bartlomieju Feb 1, 2022
b35cfa2
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 2, 2022
f034640
fix after merge
bartlomieju Feb 2, 2022
c2474f2
wip
bartlomieju Feb 2, 2022
752df11
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 3, 2022
53723e4
wip2
bartlomieju Feb 3, 2022
810ac05
recursive analysis
bartlomieju Feb 3, 2022
898cf1d
more progress
bartlomieju Feb 4, 2022
703da80
almost working
bartlomieju Feb 4, 2022
284f4f9
working!
bartlomieju Feb 4, 2022
750fca2
fmt
bartlomieju Feb 4, 2022
4da573b
lint
bartlomieju Feb 4, 2022
2d3f690
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 5, 2022
daaa9ab
simplify test
bartlomieju Feb 5, 2022
cf9142c
move files to cli/tests/
bartlomieju Feb 5, 2022
f31d9b2
remove not needed file
bartlomieju Feb 5, 2022
1a19d6a
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 6, 2022
56301eb
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 6, 2022
061b8f8
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 15, 2022
00fc975
use node_resolve crate
bartlomieju Feb 15, 2022
4f049e8
remove debug log
bartlomieju Feb 15, 2022
34d93f7
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 21, 2022
57aa65b
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 22, 2022
66658c6
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 24, 2022
bf63bab
use node_resolver
bartlomieju Feb 24, 2022
6b0e3bd
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 24, 2022
46b97b4
update std submodule
bartlomieju Feb 24, 2022
5e9c219
Merge branch 'main' into cjs_esm_integration
bartlomieju Feb 26, 2022
fc827bb
add test
bartlomieju Feb 26, 2022
9e75822
update test
bartlomieju Feb 26, 2022
7ea9d72
add missing newline :D
bartlomieju Feb 26, 2022
604289f
use --queit
bartlomieju Feb 26, 2022
cbcf804
escape file path
bartlomieju Feb 26, 2022
8b792b1
move to compat/mod.rs
bartlomieju Feb 27, 2022
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
12 changes: 12 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 2 additions & 1 deletion cli/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ winapi = "=0.3.9"
winres = "=0.1.11"

[dependencies]
deno_ast = { version = "0.12.0", features = ["bundler", "codegen", "dep_graph", "module_specifier", "proposal", "react", "sourcemap", "transforms", "transpiling", "typescript", "view", "visit"] }
deno_ast = { version = "0.12.0", features = ["bundler", "cjs", "codegen", "dep_graph", "module_specifier", "proposal", "react", "sourcemap", "transforms", "transpiling", "typescript", "view", "visit"] }
deno_core = { version = "0.120.0", path = "../core" }
deno_doc = "0.32.0"
deno_graph = "0.24.0"
Expand Down Expand Up @@ -74,6 +74,7 @@ jsonc-parser = { version = "=0.19.0", features = ["serde"] }
libc = "=0.2.106"
log = { version = "=0.4.14", features = ["serde"] }
lspower = "=1.4.0"
node_resolver = "0.1.0"
notify = "=5.0.0-pre.12"
once_cell = "=1.9.0"
percent-encoding = "=2.1.0"
Expand Down
97 changes: 97 additions & 0 deletions cli/compat/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,11 +3,15 @@
mod errors;
mod esm_resolver;

use crate::file_fetcher::FileFetcher;
use deno_ast::MediaType;
use deno_core::error::AnyError;
use deno_core::located_script_name;
use deno_core::url::Url;
use deno_core::JsRuntime;
use deno_core::ModuleSpecifier;
use once_cell::sync::Lazy;
use std::sync::Arc;

pub use esm_resolver::check_if_should_use_esm_loader;
pub(crate) use esm_resolver::NodeEsmResolver;
Expand Down Expand Up @@ -155,3 +159,96 @@ pub fn setup_builtin_modules(
js_runtime.execute_script("setup_node_builtins.js", &script)?;
Ok(())
}

/// Translates given CJS module into ESM. This function will perform static
/// analysis on the file to find defined exports and reexports.
///
/// For all discovered reexports the analysis will be performed recursively.
///
/// If successful a source code for equivalent ES module is returned.
pub async fn translate_cjs_to_esm(
file_fetcher: &FileFetcher,
specifier: &ModuleSpecifier,
code: String,
media_type: MediaType,
) -> Result<String, AnyError> {
let parsed_source = deno_ast::parse_script(deno_ast::ParseParams {
specifier: specifier.to_string(),
source: deno_ast::SourceTextInfo::new(Arc::new(code)),
media_type,
capture_tokens: true,
scope_analysis: false,
maybe_syntax: None,
})?;
let analysis = parsed_source.analyze_cjs();

let mut source = vec![
r#"import { createRequire } from "node:module";"#.to_string(),
r#"const require = createRequire(import.meta.url);"#.to_string(),
];

// if there are reexports, handle them first
for (idx, reexport) in analysis.reexports.iter().enumerate() {
// Firstly, resolve relate reexport specifier
let resolved_reexport = node_resolver::node_resolve(
reexport,
&specifier.to_file_path().unwrap(),
// FIXME(bartlomieju): check if these conditions are okay, probably
// should be `deno-require`, because `deno` is already used in `esm_resolver.rs`
&["deno", "require", "default"],
)?;
let reexport_specifier =
ModuleSpecifier::from_file_path(&resolved_reexport).unwrap();
// Secondly, read the source code from disk
let reexport_file = file_fetcher.get_source(&reexport_specifier).unwrap();
// Now perform analysis again
{
let parsed_source = deno_ast::parse_script(deno_ast::ParseParams {
specifier: reexport_specifier.to_string(),
source: deno_ast::SourceTextInfo::new(reexport_file.source),
media_type: reexport_file.media_type,
capture_tokens: true,
scope_analysis: false,
maybe_syntax: None,
})?;
let analysis = parsed_source.analyze_cjs();

source.push(format!(
"const reexport{} = require(\"{}\");",
idx, reexport
));

for export in analysis.exports.iter().filter(|e| e.as_str() != "default")
{
// TODO(bartlomieju): Node actually checks if a given export exists in `exports` object,
// but it might not be necessary here since our analysis is more detailed?
source.push(format!(
"export const {} = reexport{}.{};",
export, idx, export
));
}
}
}

source.push(format!(
"const mod = require(\"{}\");",
specifier
.to_file_path()
.unwrap()
.to_str()
.unwrap()
.replace('\\', "\\\\")
.replace('\'', "\\\'")
.replace('\"', "\\\"")
));
source.push("export default mod".to_string());

for export in analysis.exports.iter().filter(|e| e.as_str() != "default") {
// TODO(bartlomieju): Node actually checks if a given export exists in `exports` object,
// but it might not be necessary here since our analysis is more detailed?
source.push(format!("export const {} = mod.{};", export, export));
}

let translated_source = source.join("\n");
Ok(translated_source)
}
23 changes: 23 additions & 0 deletions cli/graph_util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ pub(crate) struct GraphData {
/// error messages.
referrer_map: HashMap<ModuleSpecifier, Range>,
configurations: HashSet<ModuleSpecifier>,
cjs_esm_translations: HashMap<ModuleSpecifier, String>,
}

impl GraphData {
Expand Down Expand Up @@ -254,6 +255,7 @@ impl GraphData {
modules,
referrer_map,
configurations: self.configurations.clone(),
cjs_esm_translations: Default::default(),
})
}

Expand Down Expand Up @@ -412,6 +414,27 @@ impl GraphData {
) -> Option<&'a ModuleEntry> {
self.modules.get(specifier)
}

// TODO(bartlomieju): after saving translated source
// it's never removed, potentially leading to excessive
// memory consumption
pub(crate) fn add_cjs_esm_translation(
&mut self,
specifier: &ModuleSpecifier,
source: String,
) {
let prev = self
.cjs_esm_translations
.insert(specifier.to_owned(), source);
assert!(prev.is_none());
}

pub(crate) fn get_cjs_esm_translation<'a>(
&'a self,
specifier: &ModuleSpecifier,
) -> Option<&'a String> {
self.cjs_esm_translations.get(specifier)
}
}

impl From<&ModuleGraph> for GraphData {
Expand Down
37 changes: 36 additions & 1 deletion cli/proc_state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -340,6 +340,34 @@ impl ProcState {
None,
)
.await;

let needs_cjs_esm_translation = graph
.modules()
.iter()
.any(|m| m.kind == ModuleKind::CommonJs);

if needs_cjs_esm_translation {
for module in graph.modules() {
// TODO(bartlomieju): this is overly simplistic heuristic, once we are
// in compat mode, all files ending with plain `.js` extension are
// considered CommonJs modules. Which leads to situation where valid
// ESM modules with `.js` extension might undergo translation (it won't
// work in this situation).
if module.kind == ModuleKind::CommonJs {
bartlomieju marked this conversation as resolved.
Show resolved Hide resolved
let translated_source = compat::translate_cjs_to_esm(
&self.file_fetcher,
&module.specifier,
module.maybe_source.as_ref().unwrap().to_string(),
module.media_type,
)
.await?;
let mut graph_data = self.graph_data.write();
graph_data
.add_cjs_esm_translation(&module.specifier, translated_source);
}
}
}

// If there was a locker, validate the integrity of all the modules in the
// locker.
graph_lock_or_exit(&graph);
Expand Down Expand Up @@ -506,7 +534,14 @@ impl ProcState {
| MediaType::Unknown
| MediaType::Cjs
| MediaType::Mjs
| MediaType::Json => code.as_ref().clone(),
| MediaType::Json => {
if let Some(source) = graph_data.get_cjs_esm_translation(&specifier)
{
source.to_owned()
} else {
code.as_ref().clone()
}
}
MediaType::Dts => "".to_string(),
_ => {
let emit_path = self
Expand Down
6 changes: 6 additions & 0 deletions cli/tests/integration/compat_tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,12 @@ itest!(compat_worker {
output: "compat/worker/worker_test.out",
});

itest!(cjs_esm_interop {
args:
"run --compat --unstable -A --quiet --no-check compat/import_cjs_from_esm/main.mjs",
output: "compat/import_cjs_from_esm.out",
});

#[test]
fn globals_in_repl() {
let (out, _err) = util::run_and_collect_output_with_args(
Expand Down
1 change: 1 addition & 0 deletions cli/tests/testdata/compat/import_cjs_from_esm.out
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
{ a: "A", b: "B", foo: "foo", bar: "bar", fizz: { buzz: "buzz", fizz: "FIZZ" } }
9 changes: 9 additions & 0 deletions cli/tests/testdata/compat/import_cjs_from_esm/imported.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
exports = {
a: "A",
b: "B",
};
exports.foo = "foo";
exports.bar = "bar";
exports.fizz = require("./reexports.js");

console.log(exports);
1 change: 1 addition & 0 deletions cli/tests/testdata/compat/import_cjs_from_esm/main.mjs
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
import "./imported.js";
1 change: 1 addition & 0 deletions cli/tests/testdata/compat/import_cjs_from_esm/reexports.js
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
module.exports = require("./reexports2.js");
2 changes: 2 additions & 0 deletions cli/tests/testdata/compat/import_cjs_from_esm/reexports2.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
exports.buzz = "buzz";
exports.fizz = "FIZZ";