swc/crates/swc_plugin_runner/tests/ecma_integration.rs
OJ Kwon ac5ab607c9
refactor(plugin/runner): Revise cache, module loading (#7408)
**Description:**

One of the oversight around design of `TransformExecutor` is
encapsulating plugin module logic. It has access to the cache and do its
own loading & storing. This means consumer of plugin runner have tricky
challenge to control its caching system. First, there is no way to
escape how swc_plugin_runner controls cache and cannot synchronize into
their own, also depends on the usecases cannot control the features they
want to opt in: for example, there's no way one interface uses in-memory
cache, and another uses filesystem since it is compile time configured
singleton.

PR revisits overall design of TransformExecutor: now it accepts a tratir
`PluginModuleBytes`, which abstracts any kind of bytes we are dealing
with, such as raw file slice or serialized `wasmer::Module`. Cache
instantiation and managing is now bubbled up to the application level
(`swc` in here), so if someone wants non-singleton caching or integrate
into their own caching system it can be customized.

Lastly, deprecated `memory_cache` feature and only exposes
`filesystem_cache`. Cache implementation uses in-memory is always
available, and can opt in filesystem cache where it's supported.


**BREAKING CHANGE:**

This is clearly breaking changes for the consumers of swc_core. for the
@swc/core, this PR takes care of necessary changes. I'll work on
next-swc changes later once we have new @swc/core version with this
changes.
2023-05-18 07:05:39 +00:00

292 lines
9.2 KiB
Rust

#![cfg_attr(not(feature = "__rkyv"), allow(warnings))]
use std::{
env, fs,
path::{Path, PathBuf},
process::{Command, Stdio},
sync::Arc,
};
use anyhow::{anyhow, Error};
use serde_json::json;
#[cfg(feature = "__rkyv")]
use swc_common::plugin::serialized::PluginSerializedBytes;
use swc_common::{
collections::AHashMap, errors::HANDLER, plugin::metadata::TransformPluginMetadataContext,
sync::Lazy, FileName, Mark,
};
use swc_ecma_ast::{CallExpr, Callee, EsVersion, Expr, Lit, MemberExpr, Program, Str};
use swc_ecma_parser::{parse_file_as_program, Syntax};
use swc_ecma_visit::{Visit, VisitWith};
/// Returns the path to the built plugin
fn build_plugin(dir: &Path) -> Result<PathBuf, Error> {
{
let mut cmd = Command::new("cargo");
cmd.current_dir(dir);
cmd.args(["build", "--target=wasm32-wasi"])
.stderr(Stdio::inherit());
cmd.output()?;
if !cmd
.status()
.expect("Exit code should be available")
.success()
{
return Err(anyhow!("Failed to build plugin"));
}
}
for entry in fs::read_dir(&dir.join("target").join("wasm32-wasi").join("debug"))? {
let entry = entry?;
let s = entry.file_name().to_string_lossy().into_owned();
if s.eq_ignore_ascii_case("swc_internal_plugin.wasm") {
return Ok(entry.path());
}
}
Err(anyhow!("Could not find built plugin"))
}
struct TestVisitor {
pub plugin_transform_found: bool,
}
impl Visit for TestVisitor {
fn visit_call_expr(&mut self, call: &CallExpr) {
if let Callee::Expr(expr) = &call.callee {
if let Expr::Member(MemberExpr { obj, .. }) = &**expr {
if let Expr::Ident(ident) = &**obj {
if ident.sym == *"console" {
let args = &*(call.args[0].expr);
if let Expr::Lit(Lit::Str(Str { value, .. })) = args {
self.plugin_transform_found = value == "changed_via_plugin";
}
}
}
}
}
}
}
#[cfg(feature = "__rkyv")]
static PLUGIN_BYTES: Lazy<swc_plugin_runner::plugin_module_bytes::CompiledPluginModuleBytes> =
Lazy::new(|| {
let path = build_plugin(
&PathBuf::from(env::var("CARGO_MANIFEST_DIR").unwrap())
.join("tests")
.join("fixture")
.join("swc_internal_plugin"),
)
.unwrap();
let raw_module_bytes = std::fs::read(&path).expect("Should able to read plugin bytes");
let store = wasmer::Store::default();
let module = wasmer::Module::new(&store, raw_module_bytes).unwrap();
swc_plugin_runner::plugin_module_bytes::CompiledPluginModuleBytes::new(
path.as_os_str()
.to_str()
.expect("Should able to get path")
.to_string(),
module,
store,
)
});
#[cfg(feature = "__rkyv")]
#[test]
fn internal() -> Result<(), Error> {
use swc_common::plugin::serialized::VersionedSerializable;
// run single plugin
testing::run_test(false, |cm, _handler| {
let fm = cm.new_source_file(FileName::Anon, "console.log(foo)".into());
let program = parse_file_as_program(
&fm,
Syntax::Es(Default::default()),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let program = PluginSerializedBytes::try_serialize(&VersionedSerializable::new(program))
.expect("Should serializable");
let experimental_metadata: AHashMap<String, String> = [
(
"TestExperimental".to_string(),
"ExperimentalValue".to_string(),
),
("OtherTest".to_string(), "OtherVal".to_string()),
]
.into_iter()
.collect();
let mut plugin_transform_executor = swc_plugin_runner::create_plugin_transform_executor(
&cm,
&Mark::new(),
&Arc::new(TransformPluginMetadataContext::new(
None,
"development".to_string(),
Some(experimental_metadata),
)),
Box::new(PLUGIN_BYTES.clone()),
Some(json!({ "pluginConfig": "testValue" })),
);
/* [TODO]: reenable this later
assert!(!plugin_transform_executor
.plugin_core_diag
.pkg_version
.is_empty());
*/
let program_bytes = plugin_transform_executor
.transform(&program, Some(false))
.expect("Plugin should apply transform");
let program: Program = program_bytes
.deserialize()
.expect("Should able to deserialize")
.into_inner();
let mut visitor = TestVisitor {
plugin_transform_found: false,
};
program.visit_with(&mut visitor);
visitor
.plugin_transform_found
.then(|| visitor.plugin_transform_found)
.ok_or(())
})
.expect("Should able to run single plugin transform");
// run single plugin with handler
testing::run_test2(false, |cm, handler| {
let fm = cm.new_source_file(FileName::Anon, "console.log(foo)".into());
let program = parse_file_as_program(
&fm,
Syntax::Es(Default::default()),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let program = PluginSerializedBytes::try_serialize(&VersionedSerializable::new(program))
.expect("Should serializable");
let experimental_metadata: AHashMap<String, String> = [
(
"TestExperimental".to_string(),
"ExperimentalValue".to_string(),
),
("OtherTest".to_string(), "OtherVal".to_string()),
]
.into_iter()
.collect();
let _res = HANDLER.set(&handler, || {
let mut plugin_transform_executor = swc_plugin_runner::create_plugin_transform_executor(
&cm,
&Mark::new(),
&Arc::new(TransformPluginMetadataContext::new(
None,
"development".to_string(),
Some(experimental_metadata),
)),
Box::new(PLUGIN_BYTES.clone()),
Some(json!({ "pluginConfig": "testValue" })),
);
plugin_transform_executor
.transform(&program, Some(false))
.expect("Plugin should apply transform")
});
Ok(())
})
.expect("Should able to run single plugin transform with handler");
// Run multiple plugins.
testing::run_test(false, |cm, _handler| {
let fm = cm.new_source_file(FileName::Anon, "console.log(foo)".into());
let program = parse_file_as_program(
&fm,
Syntax::Es(Default::default()),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let mut serialized_program =
PluginSerializedBytes::try_serialize(&VersionedSerializable::new(program))
.expect("Should serializable");
let experimental_metadata: AHashMap<String, String> = [
(
"TestExperimental".to_string(),
"ExperimentalValue".to_string(),
),
("OtherTest".to_string(), "OtherVal".to_string()),
]
.into_iter()
.collect();
let mut plugin_transform_executor = swc_plugin_runner::create_plugin_transform_executor(
&cm,
&Mark::new(),
&Arc::new(TransformPluginMetadataContext::new(
None,
"development".to_string(),
Some(experimental_metadata.clone()),
)),
Box::new(PLUGIN_BYTES.clone()),
Some(json!({ "pluginConfig": "testValue" })),
);
serialized_program = plugin_transform_executor
.transform(&serialized_program, Some(false))
.expect("Plugin should apply transform");
// TODO: we'll need to apply 2 different plugins
let mut plugin_transform_executor = swc_plugin_runner::create_plugin_transform_executor(
&cm,
&Mark::new(),
&Arc::new(TransformPluginMetadataContext::new(
None,
"development".to_string(),
Some(experimental_metadata),
)),
Box::new(PLUGIN_BYTES.clone()),
Some(json!({ "pluginConfig": "testValue" })),
);
serialized_program = plugin_transform_executor
.transform(&serialized_program, Some(false))
.expect("Plugin should apply transform");
let program: Program = serialized_program
.deserialize()
.expect("Should able to deserialize")
.into_inner();
let mut visitor = TestVisitor {
plugin_transform_found: false,
};
program.visit_with(&mut visitor);
visitor
.plugin_transform_found
.then(|| visitor.plugin_transform_found)
.ok_or(())
})
.expect("Should able to run multiple plugins transform");
Ok(())
}