mirror of
https://github.com/denoland/deno.git
synced 2024-11-21 15:04:11 -05:00
refactor(test): grab runTests() and runBenchmarks() from __bootstrap (#15420)
This commit is contained in:
parent
5262937285
commit
4f8dea100e
5 changed files with 128 additions and 64 deletions
|
@ -38,12 +38,12 @@ failing step in failing test ... FAILED ([WILDCARD])
|
||||||
nested failure => ./test/steps/failing_steps.ts:[WILDCARD]
|
nested failure => ./test/steps/failing_steps.ts:[WILDCARD]
|
||||||
error: Error: 1 test step failed.
|
error: Error: 1 test step failed.
|
||||||
at runTest (deno:runtime/js/40_testing.js:[WILDCARD])
|
at runTest (deno:runtime/js/40_testing.js:[WILDCARD])
|
||||||
at async Object.runTests (deno:runtime/js/40_testing.js:[WILDCARD])
|
at async runTests (deno:runtime/js/40_testing.js:[WILDCARD])
|
||||||
|
|
||||||
multiple test step failures => ./test/steps/failing_steps.ts:[WILDCARD]
|
multiple test step failures => ./test/steps/failing_steps.ts:[WILDCARD]
|
||||||
error: Error: 2 test steps failed.
|
error: Error: 2 test steps failed.
|
||||||
at runTest (deno:runtime/js/40_testing.js:[WILDCARD])
|
at runTest (deno:runtime/js/40_testing.js:[WILDCARD])
|
||||||
at async Object.runTests (deno:runtime/js/40_testing.js:[WILDCARD])
|
at async runTests (deno:runtime/js/40_testing.js:[WILDCARD])
|
||||||
|
|
||||||
failing step in failing test => ./test/steps/failing_steps.ts:[WILDCARD]
|
failing step in failing test => ./test/steps/failing_steps.ts:[WILDCARD]
|
||||||
error: Error: Fail test.
|
error: Error: Fail test.
|
||||||
|
|
|
@ -7,7 +7,6 @@ use deno_core::error::AnyError;
|
||||||
use deno_core::futures::task::LocalFutureObj;
|
use deno_core::futures::task::LocalFutureObj;
|
||||||
use deno_core::futures::FutureExt;
|
use deno_core::futures::FutureExt;
|
||||||
use deno_core::located_script_name;
|
use deno_core::located_script_name;
|
||||||
use deno_core::serde_json::json;
|
|
||||||
use deno_core::Extension;
|
use deno_core::Extension;
|
||||||
use deno_core::ModuleId;
|
use deno_core::ModuleId;
|
||||||
use deno_graph::source::ResolveResponse;
|
use deno_graph::source::ResolveResponse;
|
||||||
|
@ -220,10 +219,7 @@ impl CliMainWorker {
|
||||||
&mut self,
|
&mut self,
|
||||||
mode: TestMode,
|
mode: TestMode,
|
||||||
) -> Result<(), AnyError> {
|
) -> Result<(), AnyError> {
|
||||||
self.worker.js_runtime.execute_script(
|
self.worker.enable_test();
|
||||||
&located_script_name!(),
|
|
||||||
r#"Deno[Deno.internal].enableTestAndBench()"#,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
// Enable op call tracing in core to enable better debugging of op sanitizer
|
// Enable op call tracing in core to enable better debugging of op sanitizer
|
||||||
// failures.
|
// failures.
|
||||||
|
@ -273,17 +269,10 @@ impl CliMainWorker {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.worker.dispatch_load_event(&located_script_name!())?;
|
self.worker.dispatch_load_event(&located_script_name!())?;
|
||||||
|
self
|
||||||
let test_result = self.worker.js_runtime.execute_script(
|
.worker
|
||||||
&located_script_name!(),
|
.run_tests(&self.ps.options.shuffle_tests())
|
||||||
&format!(
|
.await?;
|
||||||
r#"Deno[Deno.internal].runTests({})"#,
|
|
||||||
json!({ "shuffle": self.ps.options.shuffle_tests() }),
|
|
||||||
),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
self.worker.js_runtime.resolve_value(test_result).await?;
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
if !self
|
if !self
|
||||||
.worker
|
.worker
|
||||||
|
@ -309,10 +298,7 @@ impl CliMainWorker {
|
||||||
&mut self,
|
&mut self,
|
||||||
mode: TestMode,
|
mode: TestMode,
|
||||||
) -> Result<(), AnyError> {
|
) -> Result<(), AnyError> {
|
||||||
self.worker.js_runtime.execute_script(
|
self.worker.enable_test();
|
||||||
&located_script_name!(),
|
|
||||||
r#"Deno[Deno.internal].enableTestAndBench()"#,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
self
|
self
|
||||||
.worker
|
.worker
|
||||||
|
@ -328,14 +314,7 @@ impl CliMainWorker {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.worker.dispatch_load_event(&located_script_name!())?;
|
self.worker.dispatch_load_event(&located_script_name!())?;
|
||||||
|
self.worker.run_tests(&None).await?;
|
||||||
let test_result = self.worker.js_runtime.execute_script(
|
|
||||||
&located_script_name!(),
|
|
||||||
r#"Deno[Deno.internal].runTests()"#,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
self.worker.js_runtime.resolve_value(test_result).await?;
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
if !self
|
if !self
|
||||||
.worker
|
.worker
|
||||||
|
@ -350,10 +329,7 @@ impl CliMainWorker {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub async fn run_bench_specifier(&mut self) -> Result<(), AnyError> {
|
pub async fn run_bench_specifier(&mut self) -> Result<(), AnyError> {
|
||||||
self.worker.js_runtime.execute_script(
|
self.worker.enable_bench();
|
||||||
&located_script_name!(),
|
|
||||||
r#"Deno[Deno.internal].enableTestAndBench()"#,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
if self.ps.options.compat() {
|
if self.ps.options.compat() {
|
||||||
self.worker.execute_side_module(&compat::GLOBAL_URL).await?;
|
self.worker.execute_side_module(&compat::GLOBAL_URL).await?;
|
||||||
|
@ -383,14 +359,7 @@ impl CliMainWorker {
|
||||||
}
|
}
|
||||||
|
|
||||||
self.worker.dispatch_load_event(&located_script_name!())?;
|
self.worker.dispatch_load_event(&located_script_name!())?;
|
||||||
|
self.worker.run_benchmarks().await?;
|
||||||
let bench_result = self.worker.js_runtime.execute_script(
|
|
||||||
&located_script_name!(),
|
|
||||||
r#"Deno[Deno.internal].runBenchmarks()"#,
|
|
||||||
)?;
|
|
||||||
|
|
||||||
self.worker.js_runtime.resolve_value(bench_result).await?;
|
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
if !self
|
if !self
|
||||||
.worker
|
.worker
|
||||||
|
|
|
@ -629,7 +629,7 @@ impl JsRuntime {
|
||||||
.ok()
|
.ok()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn grab_global<'s, T>(
|
pub fn grab_global<'s, T>(
|
||||||
scope: &mut v8::HandleScope<'s>,
|
scope: &mut v8::HandleScope<'s>,
|
||||||
path: &str,
|
path: &str,
|
||||||
) -> Option<v8::Local<'s, T>>
|
) -> Option<v8::Local<'s, T>>
|
||||||
|
|
|
@ -16,6 +16,7 @@
|
||||||
ArrayPrototypePush,
|
ArrayPrototypePush,
|
||||||
ArrayPrototypeShift,
|
ArrayPrototypeShift,
|
||||||
ArrayPrototypeSort,
|
ArrayPrototypeSort,
|
||||||
|
BigInt,
|
||||||
DateNow,
|
DateNow,
|
||||||
Error,
|
Error,
|
||||||
FunctionPrototype,
|
FunctionPrototype,
|
||||||
|
@ -600,7 +601,8 @@
|
||||||
const testStates = new Map();
|
const testStates = new Map();
|
||||||
/** @type {BenchDescription[]} */
|
/** @type {BenchDescription[]} */
|
||||||
const benchDescs = [];
|
const benchDescs = [];
|
||||||
let isTestOrBenchSubcommand = false;
|
let isTestSubcommand = false;
|
||||||
|
let isBenchSubcommand = false;
|
||||||
|
|
||||||
// Main test function provided by Deno.
|
// Main test function provided by Deno.
|
||||||
function test(
|
function test(
|
||||||
|
@ -608,7 +610,7 @@
|
||||||
optionsOrFn,
|
optionsOrFn,
|
||||||
maybeFn,
|
maybeFn,
|
||||||
) {
|
) {
|
||||||
if (!isTestOrBenchSubcommand) {
|
if (!isTestSubcommand) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -727,7 +729,7 @@
|
||||||
optionsOrFn,
|
optionsOrFn,
|
||||||
maybeFn,
|
maybeFn,
|
||||||
) {
|
) {
|
||||||
if (!isTestOrBenchSubcommand) {
|
if (!isBenchSubcommand) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1032,11 +1034,12 @@
|
||||||
return ops.op_bench_now();
|
return ops.op_bench_now();
|
||||||
}
|
}
|
||||||
|
|
||||||
// This function is called by Rust side if we're in `deno test` or
|
function enableTest() {
|
||||||
// `deno bench` subcommand. If this function is not called then `Deno.test()`
|
isTestSubcommand = true;
|
||||||
// and `Deno.bench()` become noops.
|
}
|
||||||
function enableTestAndBench() {
|
|
||||||
isTestOrBenchSubcommand = true;
|
function enableBench() {
|
||||||
|
isBenchSubcommand = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function runTests({
|
async function runTests({
|
||||||
|
@ -1062,15 +1065,16 @@
|
||||||
|
|
||||||
if (shuffle !== null) {
|
if (shuffle !== null) {
|
||||||
// http://en.wikipedia.org/wiki/Linear_congruential_generator
|
// http://en.wikipedia.org/wiki/Linear_congruential_generator
|
||||||
|
// Use BigInt for everything because the random seed is u64.
|
||||||
const nextInt = (function (state) {
|
const nextInt = (function (state) {
|
||||||
const m = 0x80000000;
|
const m = 0x80000000n;
|
||||||
const a = 1103515245;
|
const a = 1103515245n;
|
||||||
const c = 12345;
|
const c = 12345n;
|
||||||
|
|
||||||
return function (max) {
|
return function (max) {
|
||||||
return state = ((a * state + c) % m) % max;
|
return state = ((a * state + c) % m) % BigInt(max);
|
||||||
};
|
};
|
||||||
}(shuffle));
|
}(BigInt(shuffle)));
|
||||||
|
|
||||||
for (let i = filtered.length - 1; i > 0; i--) {
|
for (let i = filtered.length - 1; i > 0; i--) {
|
||||||
const j = nextInt(i);
|
const j = nextInt(i);
|
||||||
|
@ -1390,15 +1394,12 @@
|
||||||
return testFn;
|
return testFn;
|
||||||
}
|
}
|
||||||
|
|
||||||
window.__bootstrap.internals = {
|
|
||||||
...window.__bootstrap.internals ?? {},
|
|
||||||
enableTestAndBench,
|
|
||||||
runTests,
|
|
||||||
runBenchmarks,
|
|
||||||
};
|
|
||||||
|
|
||||||
window.__bootstrap.testing = {
|
window.__bootstrap.testing = {
|
||||||
test,
|
|
||||||
bench,
|
bench,
|
||||||
|
enableBench,
|
||||||
|
enableTest,
|
||||||
|
runBenchmarks,
|
||||||
|
runTests,
|
||||||
|
test,
|
||||||
};
|
};
|
||||||
})(this);
|
})(this);
|
||||||
|
|
|
@ -11,6 +11,9 @@ use deno_core::error::AnyError;
|
||||||
use deno_core::error::JsError;
|
use deno_core::error::JsError;
|
||||||
use deno_core::futures::Future;
|
use deno_core::futures::Future;
|
||||||
use deno_core::located_script_name;
|
use deno_core::located_script_name;
|
||||||
|
use deno_core::serde_json::json;
|
||||||
|
use deno_core::serde_v8;
|
||||||
|
use deno_core::v8;
|
||||||
use deno_core::CompiledWasmModuleStore;
|
use deno_core::CompiledWasmModuleStore;
|
||||||
use deno_core::Extension;
|
use deno_core::Extension;
|
||||||
use deno_core::GetErrorClassFn;
|
use deno_core::GetErrorClassFn;
|
||||||
|
@ -59,6 +62,10 @@ pub struct MainWorker {
|
||||||
pub js_runtime: JsRuntime,
|
pub js_runtime: JsRuntime,
|
||||||
should_break_on_first_statement: bool,
|
should_break_on_first_statement: bool,
|
||||||
exit_code: ExitCode,
|
exit_code: ExitCode,
|
||||||
|
js_run_tests_callback: v8::Global<v8::Function>,
|
||||||
|
js_run_benchmarks_callback: v8::Global<v8::Function>,
|
||||||
|
js_enable_test_callback: v8::Global<v8::Function>,
|
||||||
|
js_enable_bench_callback: v8::Global<v8::Function>,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct WorkerOptions {
|
pub struct WorkerOptions {
|
||||||
|
@ -86,6 +93,15 @@ pub struct WorkerOptions {
|
||||||
pub stdio: Stdio,
|
pub stdio: Stdio,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn grab_cb(
|
||||||
|
scope: &mut v8::HandleScope,
|
||||||
|
path: &str,
|
||||||
|
) -> v8::Global<v8::Function> {
|
||||||
|
let cb = JsRuntime::grab_global::<v8::Function>(scope, path)
|
||||||
|
.unwrap_or_else(|| panic!("{} must be defined", path));
|
||||||
|
v8::Global::new(scope, cb)
|
||||||
|
}
|
||||||
|
|
||||||
impl MainWorker {
|
impl MainWorker {
|
||||||
pub fn bootstrap_from_options(
|
pub fn bootstrap_from_options(
|
||||||
main_module: ModuleSpecifier,
|
main_module: ModuleSpecifier,
|
||||||
|
@ -198,10 +214,29 @@ impl MainWorker {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let (
|
||||||
|
js_run_tests_callback,
|
||||||
|
js_run_benchmarks_callback,
|
||||||
|
js_enable_test_callback,
|
||||||
|
js_enable_bench_callback,
|
||||||
|
) = {
|
||||||
|
let scope = &mut js_runtime.handle_scope();
|
||||||
|
(
|
||||||
|
grab_cb(scope, "__bootstrap.testing.runTests"),
|
||||||
|
grab_cb(scope, "__bootstrap.testing.runBenchmarks"),
|
||||||
|
grab_cb(scope, "__bootstrap.testing.enableTest"),
|
||||||
|
grab_cb(scope, "__bootstrap.testing.enableBench"),
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
js_runtime,
|
js_runtime,
|
||||||
should_break_on_first_statement: options.should_break_on_first_statement,
|
should_break_on_first_statement: options.should_break_on_first_statement,
|
||||||
exit_code,
|
exit_code,
|
||||||
|
js_run_tests_callback,
|
||||||
|
js_run_benchmarks_callback,
|
||||||
|
js_enable_test_callback,
|
||||||
|
js_enable_bench_callback,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -289,6 +324,65 @@ impl MainWorker {
|
||||||
self.evaluate_module(id).await
|
self.evaluate_module(id).await
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Run tests declared with `Deno.test()`. Test events will be dispatched
|
||||||
|
/// by calling ops which are currently only implemented in the CLI crate.
|
||||||
|
// TODO(nayeemrmn): Move testing ops to deno_runtime and redesign/unhide.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub async fn run_tests(
|
||||||
|
&mut self,
|
||||||
|
shuffle: &Option<u64>,
|
||||||
|
) -> Result<(), AnyError> {
|
||||||
|
let promise = {
|
||||||
|
let scope = &mut self.js_runtime.handle_scope();
|
||||||
|
let cb = self.js_run_tests_callback.open(scope);
|
||||||
|
let this = v8::undefined(scope).into();
|
||||||
|
let options =
|
||||||
|
serde_v8::to_v8(scope, json!({ "shuffle": shuffle })).unwrap();
|
||||||
|
let promise = cb.call(scope, this, &[options]).unwrap();
|
||||||
|
v8::Global::new(scope, promise)
|
||||||
|
};
|
||||||
|
self.js_runtime.resolve_value(promise).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Run benches declared with `Deno.bench()`. Bench events will be dispatched
|
||||||
|
/// by calling ops which are currently only implemented in the CLI crate.
|
||||||
|
// TODO(nayeemrmn): Move benchmark ops to deno_runtime and redesign/unhide.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub async fn run_benchmarks(&mut self) -> Result<(), AnyError> {
|
||||||
|
let promise = {
|
||||||
|
let scope = &mut self.js_runtime.handle_scope();
|
||||||
|
let cb = self.js_run_benchmarks_callback.open(scope);
|
||||||
|
let this = v8::undefined(scope).into();
|
||||||
|
let promise = cb.call(scope, this, &[]).unwrap();
|
||||||
|
v8::Global::new(scope, promise)
|
||||||
|
};
|
||||||
|
self.js_runtime.resolve_value(promise).await?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable `Deno.test()`. If this isn't called before executing user code,
|
||||||
|
/// `Deno.test()` calls will noop.
|
||||||
|
// TODO(nayeemrmn): Move testing ops to deno_runtime and redesign/unhide.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub fn enable_test(&mut self) {
|
||||||
|
let scope = &mut self.js_runtime.handle_scope();
|
||||||
|
let cb = self.js_enable_test_callback.open(scope);
|
||||||
|
let this = v8::undefined(scope).into();
|
||||||
|
cb.call(scope, this, &[]).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Enable `Deno.bench()`. If this isn't called before executing user code,
|
||||||
|
/// `Deno.bench()` calls will noop.
|
||||||
|
// TODO(nayeemrmn): Move benchmark ops to deno_runtime and redesign/unhide.
|
||||||
|
#[doc(hidden)]
|
||||||
|
pub fn enable_bench(&mut self) {
|
||||||
|
let scope = &mut self.js_runtime.handle_scope();
|
||||||
|
let cb = self.js_enable_bench_callback.open(scope);
|
||||||
|
let this = v8::undefined(scope).into();
|
||||||
|
cb.call(scope, this, &[]).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
fn wait_for_inspector_session(&mut self) {
|
fn wait_for_inspector_session(&mut self) {
|
||||||
if self.should_break_on_first_statement {
|
if self.should_break_on_first_statement {
|
||||||
self
|
self
|
||||||
|
|
Loading…
Reference in a new issue