swc/crates/swc_plugin_runner/tests/integration.rs
2022-02-26 06:41:14 +00:00

210 lines
6.7 KiB
Rust

use std::{
env, fs,
path::{Path, PathBuf},
process::{Command, Stdio},
};
use anyhow::{anyhow, Error};
use swc_common::{errors::HANDLER, plugin::Serialized, sync::Lazy, FileName};
use swc_ecma_ast::{CallExpr, Callee, EsVersion, Expr, Lit, MemberExpr, Program, Str};
use swc_ecma_parser::{parse_file_as_program, EsConfig, Syntax};
use swc_ecma_visit::{Visit, VisitWith};
use swc_plugin_runner::cache::PluginModuleCache;
/// 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()?;
}
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";
}
}
}
}
}
}
}
#[test]
fn internal() -> Result<(), Error> {
let path = build_plugin(
&PathBuf::from(env::var("CARGO_MANIFEST_DIR")?)
.join("..")
.join("..")
.join("tests")
.join("rust-plugins")
.join("swc_internal_plugin"),
)?;
// 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(EsConfig {
..Default::default()
}),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let program = Serialized::serialize(&program).expect("Should serializable");
let config = Serialized::serialize(&"{}".to_string()).expect("Should serializable");
let context = Serialized::serialize(&"{sourceFileName: 'single_plugin_test'}".to_string())
.expect("Should serializable");
let cache: Lazy<PluginModuleCache> = Lazy::new(PluginModuleCache::new);
let program_bytes = swc_plugin_runner::apply_js_plugin(
"internal-test",
&path,
&cache,
program,
config,
context,
)
.expect("Plugin should apply transform");
let program: Program =
Serialized::deserialize(&program_bytes).expect("Should able to deserialize");
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(EsConfig {
..Default::default()
}),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let program = Serialized::serialize(&program).expect("Should serializable");
let config = Serialized::serialize(&"{}".to_string()).expect("Should serializable");
let context =
Serialized::serialize(&"{sourceFileName: 'single_plugin_handler_test'}".to_string())
.expect("Should serializable");
let cache: Lazy<PluginModuleCache> = Lazy::new(PluginModuleCache::new);
let _res = HANDLER.set(&handler, || {
swc_plugin_runner::apply_js_plugin(
"internal-test",
&path,
&cache,
program,
config,
context,
)
.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(EsConfig {
..Default::default()
}),
EsVersion::latest(),
None,
&mut vec![],
)
.unwrap();
let mut serialized_program = Serialized::serialize(&program).expect("Should serializable");
let cache: Lazy<PluginModuleCache> = Lazy::new(PluginModuleCache::new);
serialized_program = swc_plugin_runner::apply_js_plugin(
"internal-test",
&path,
&cache,
serialized_program,
Serialized::serialize(&"{}".to_string()).expect("Should serializable"),
Serialized::serialize(&"{sourceFileName: 'multiple_plugin_test'}".to_string())
.expect("Should serializable"),
)
.expect("Plugin should apply transform");
// TODO: we'll need to apply 2 different plugins
serialized_program = swc_plugin_runner::apply_js_plugin(
"internal-test",
&path,
&cache,
serialized_program,
Serialized::serialize(&"{}".to_string()).expect("Should serializable"),
Serialized::serialize(&"{sourceFileName: 'multiple_plugin_test2'}".to_string())
.expect("Should serializable"),
)
.expect("Plugin should apply transform");
let program: Program =
Serialized::deserialize(&serialized_program).expect("Should able to deserialize");
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(())
}