2018-07-26 18:49:20 -07:00
|
|
|
//! A "wrapper binary" used to execute wasm files as tests
|
|
|
|
//!
|
|
|
|
//! This binary is intended to be used as a "test runner" for wasm binaries,
|
|
|
|
//! being compatible with `cargo test` for the wasm target. It will
|
|
|
|
//! automatically execute `wasm-bindgen` (or the equivalent thereof) and then
|
|
|
|
//! execute either Node.js over the tests or start a server which a browser can
|
|
|
|
//! be used to run against to execute tests. In a browser mode if `CI` is in the
|
|
|
|
//! environment then it'll also attempt headless testing, spawning the server in
|
|
|
|
//! the background and then using the WebDriver protocol to execute tests.
|
|
|
|
//!
|
|
|
|
//! For more documentation about this see the `wasm-bindgen-test` crate README
|
|
|
|
//! and source code.
|
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
extern crate curl;
|
|
|
|
extern crate env_logger;
|
2018-07-20 13:47:49 -05:00
|
|
|
#[macro_use]
|
|
|
|
extern crate failure;
|
2018-07-25 03:34:35 -07:00
|
|
|
#[macro_use]
|
|
|
|
extern crate log;
|
2018-07-20 13:47:49 -05:00
|
|
|
extern crate parity_wasm;
|
2018-07-24 11:27:06 -07:00
|
|
|
extern crate rouille;
|
2018-07-25 03:34:35 -07:00
|
|
|
extern crate serde;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate serde_derive;
|
|
|
|
#[macro_use]
|
|
|
|
extern crate serde_json;
|
2018-07-24 11:27:06 -07:00
|
|
|
extern crate wasm_bindgen_cli_support;
|
2018-07-20 13:47:49 -05:00
|
|
|
|
|
|
|
use std::env;
|
2018-07-25 16:06:47 -07:00
|
|
|
use std::fs;
|
2018-07-20 13:47:49 -05:00
|
|
|
use std::path::PathBuf;
|
2018-07-24 11:27:06 -07:00
|
|
|
use std::process;
|
2018-07-25 03:34:35 -07:00
|
|
|
use std::thread;
|
2018-07-20 13:47:49 -05:00
|
|
|
|
2018-09-26 08:26:00 -07:00
|
|
|
use failure::{Error, ResultExt};
|
|
|
|
use parity_wasm::elements::{Deserialize, Module, Section};
|
2018-07-20 13:47:49 -05:00
|
|
|
use wasm_bindgen_cli_support::Bindgen;
|
|
|
|
|
2018-10-17 19:15:09 -07:00
|
|
|
// no need for jemalloc bloat in this binary (and we don't need speed)
|
|
|
|
#[global_allocator]
|
|
|
|
static ALLOC: std::alloc::System = std::alloc::System;
|
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
mod headless;
|
2018-07-24 11:27:06 -07:00
|
|
|
mod node;
|
|
|
|
mod server;
|
2018-07-25 03:34:35 -07:00
|
|
|
mod shell;
|
2018-07-24 11:27:06 -07:00
|
|
|
|
2018-07-20 13:47:49 -05:00
|
|
|
fn main() {
|
2018-07-25 03:34:35 -07:00
|
|
|
env_logger::init();
|
2018-07-20 13:47:49 -05:00
|
|
|
let err = match rmain() {
|
|
|
|
Ok(()) => return,
|
|
|
|
Err(e) => e,
|
|
|
|
};
|
|
|
|
eprintln!("error: {}", err);
|
2018-08-01 16:15:09 -05:00
|
|
|
for cause in err.iter_causes() {
|
2018-07-20 13:47:49 -05:00
|
|
|
eprintln!("\tcaused by: {}", cause);
|
|
|
|
}
|
|
|
|
process::exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn rmain() -> Result<(), Error> {
|
|
|
|
let mut args = env::args_os().skip(1);
|
2018-07-25 03:34:35 -07:00
|
|
|
let shell = shell::Shell::new();
|
2018-07-20 13:47:49 -05:00
|
|
|
|
|
|
|
// Currently no flags are supported, and assume there's only one argument
|
|
|
|
// which is the wasm file to test. This'll want to improve over time!
|
|
|
|
let wasm_file_to_test = match args.next() {
|
|
|
|
Some(file) => PathBuf::from(file),
|
|
|
|
None => bail!("must have a file to test as first argument"),
|
|
|
|
};
|
|
|
|
|
|
|
|
// Assume a cargo-like directory layout and generate output at
|
|
|
|
// `target/wasm32-unknown-unknown/wbg-tmp/...`
|
2018-09-26 08:26:00 -07:00
|
|
|
let tmpdir = wasm_file_to_test
|
|
|
|
.parent() // chop off file name
|
|
|
|
.and_then(|p| p.parent()) // chop off `deps`
|
|
|
|
.and_then(|p| p.parent()) // chop off `debug`
|
2018-07-20 13:47:49 -05:00
|
|
|
.map(|p| p.join("wbg-tmp"))
|
2018-09-26 08:26:00 -07:00
|
|
|
.ok_or_else(|| format_err!("file to test doesn't follow the expected Cargo conventions"))?;
|
2018-07-20 13:47:49 -05:00
|
|
|
|
|
|
|
// Make sure there's no stale state from before
|
|
|
|
drop(fs::remove_dir_all(&tmpdir));
|
2018-09-26 08:26:00 -07:00
|
|
|
fs::create_dir(&tmpdir).context("creating temporary directory")?;
|
2018-07-20 13:47:49 -05:00
|
|
|
|
2018-08-01 14:19:19 -05:00
|
|
|
let module = "wasm-bindgen-test";
|
2018-07-20 13:47:49 -05:00
|
|
|
|
|
|
|
// Collect all tests that the test harness is supposed to run. We assume
|
|
|
|
// that any exported function with the prefix `__wbg_test` is a test we need
|
|
|
|
// to execute.
|
2018-09-26 08:26:00 -07:00
|
|
|
let wasm = fs::read(&wasm_file_to_test).context("failed to read wasm file")?;
|
|
|
|
let wasm = Module::deserialize(&mut &wasm[..]).context("failed to deserialize wasm module")?;
|
2018-07-24 11:27:06 -07:00
|
|
|
let mut tests = Vec::new();
|
2018-07-25 15:49:50 -07:00
|
|
|
if let Some(exports) = wasm.export_section() {
|
2018-07-20 13:47:49 -05:00
|
|
|
for export in exports.entries() {
|
|
|
|
if !export.field().starts_with("__wbg_test") {
|
2018-09-26 08:26:00 -07:00
|
|
|
continue;
|
2018-07-20 13:47:49 -05:00
|
|
|
}
|
2018-07-24 11:27:06 -07:00
|
|
|
tests.push(export.field().to_string());
|
2018-07-20 13:47:49 -05:00
|
|
|
}
|
|
|
|
}
|
2018-07-25 03:34:35 -07:00
|
|
|
|
|
|
|
// Right now there's a bug where if no tests are present then the
|
|
|
|
// `wasm-bindgen-test` runtime support isn't linked in, so just bail out
|
|
|
|
// early saying everything is ok.
|
2018-07-24 11:27:06 -07:00
|
|
|
if tests.len() == 0 {
|
|
|
|
println!("no tests to run!");
|
2018-09-26 08:26:00 -07:00
|
|
|
return Ok(());
|
2018-07-24 11:27:06 -07:00
|
|
|
}
|
2018-07-20 13:47:49 -05:00
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
// Figure out if this tests is supposed to execute in node.js or a browser.
|
|
|
|
// That's done on a per-test-binary basis with the
|
|
|
|
// `wasm_bindgen_test_configure` macro, which emits a custom section for us
|
|
|
|
// to read later on.
|
|
|
|
let mut node = true;
|
|
|
|
for section in wasm.sections() {
|
|
|
|
let custom = match section {
|
|
|
|
Section::Custom(section) => section,
|
|
|
|
_ => continue,
|
|
|
|
};
|
|
|
|
if custom.name() != "__wasm_bindgen_test_unstable" {
|
2018-09-26 08:26:00 -07:00
|
|
|
continue;
|
2018-07-25 03:34:35 -07:00
|
|
|
}
|
|
|
|
node = !custom.payload().contains(&0x01);
|
|
|
|
}
|
2018-08-02 10:30:07 -05:00
|
|
|
let headless = env::var("NO_HEADLESS").is_err();
|
2018-08-06 09:57:41 -07:00
|
|
|
let debug = env::var("WASM_BINDGEN_NO_DEBUG").is_err();
|
2018-07-20 13:47:49 -05:00
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
// Make the generated bindings available for the tests to execute against.
|
|
|
|
shell.status("Executing bindgen...");
|
2018-07-25 15:49:50 -07:00
|
|
|
let mut b = Bindgen::new();
|
2018-08-06 09:57:41 -07:00
|
|
|
b.debug(debug)
|
2018-07-24 11:27:06 -07:00
|
|
|
.nodejs(node)
|
2018-10-08 10:01:53 -07:00
|
|
|
.input_module(module, wasm)
|
2018-07-25 15:49:50 -07:00
|
|
|
.keep_debug(false)
|
2018-11-28 09:25:51 -08:00
|
|
|
.emit_start(false)
|
2018-07-25 15:49:50 -07:00
|
|
|
.generate(&tmpdir)
|
|
|
|
.context("executing `wasm-bindgen` over the wasm file")?;
|
2018-07-25 03:34:35 -07:00
|
|
|
shell.clear();
|
2018-07-25 15:49:50 -07:00
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
// If we're executing in node.js, that module will take it from here.
|
2018-07-24 11:27:06 -07:00
|
|
|
if node {
|
2018-09-26 08:26:00 -07:00
|
|
|
return node::execute(&module, &tmpdir, &args.collect::<Vec<_>>(), &tests);
|
2018-07-24 11:27:06 -07:00
|
|
|
}
|
2018-07-20 13:47:49 -05:00
|
|
|
|
2018-07-25 03:34:35 -07:00
|
|
|
// Otherwise we're executing in a browser. Spawn a server which serves up
|
|
|
|
// the local generated files over an HTTP server.
|
|
|
|
let srv = server::spawn(
|
|
|
|
&if headless {
|
|
|
|
"127.0.0.1:0".parse().unwrap()
|
|
|
|
} else {
|
|
|
|
"127.0.0.1:8000".parse().unwrap()
|
|
|
|
},
|
|
|
|
headless,
|
|
|
|
&module,
|
|
|
|
&tmpdir,
|
|
|
|
&args.collect::<Vec<_>>(),
|
|
|
|
&tests,
|
2018-11-29 12:25:10 -08:00
|
|
|
).context("failed to spawn server")?;
|
2018-07-25 03:34:35 -07:00
|
|
|
let addr = srv.server_addr();
|
|
|
|
|
|
|
|
// TODO: eventually we should provide the ability to exit at some point
|
|
|
|
// (gracefully) here, but for now this just runs forever.
|
|
|
|
if !headless {
|
2018-09-26 08:26:00 -07:00
|
|
|
println!(
|
|
|
|
"Interactive browsers tests are now available at http://{}",
|
|
|
|
addr
|
|
|
|
);
|
2018-08-02 10:30:07 -05:00
|
|
|
println!("");
|
|
|
|
println!("Note that interactive mode is enabled because `NO_HEADLESS`");
|
|
|
|
println!("is specified in the environment of this process. Once you're");
|
|
|
|
println!("done with testing you'll need to kill this server with");
|
|
|
|
println!("Ctrl-C.");
|
2018-09-26 08:26:00 -07:00
|
|
|
return Ok(srv.run());
|
2018-07-25 03:34:35 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
thread::spawn(|| srv.run());
|
|
|
|
headless::run(&addr, &shell)?;
|
|
|
|
Ok(())
|
2018-07-20 13:47:49 -05:00
|
|
|
}
|