mirror of
https://github.com/denoland/deno.git
synced 2024-12-24 16:19:12 -05:00
refactor: split integration tests from CLI (part 1) (#22308)
This PR separates integration tests from CLI tests into a new project named `cli_tests`. This is a prerequisite for an integration test runner that can work with either the CLI binary in the current project, or one that is built ahead of time. ## Background Rust does not have the concept of artifact dependencies yet (https://github.com/rust-lang/cargo/issues/9096). Because of this, the only way we can ensure a binary is built before running associated tests is by hanging tests off the crate with the binary itself. Unfortunately this means that to run those tests, you _must_ build the binary and in the case of the deno executable that might be a 10 minute wait in release mode. ## Implementation To allow for tests to run with and without the requirement that the binary is up-to-date, we split the integration tests into a project of their own. As these tests would not require the binary to build itself before being run as-is, we add a stub integration `[[test]]` target in the `cli` project that invokes these tests using `cargo test`. The stub test runner we add has `harness = false` so that we can get access to a `main` function. This `main` function's sole job is to `execvp` the command `cargo test -p deno_cli`, effectively "calling" another cargo target. This ensures that the deno executable is always correctly rebuilt before running the stub test runner from `cli`, and gets us closer to be able to run the entire integration test suite on arbitrary deno executables (and therefore split the build into multiple phases). The new `cli_tests` project lives within `cli` to avoid a large PR. In later PRs, the test data will be split from the `cli` project. As there are a few thousand files, it'll be better to do this as a completely separate PR to avoid noise.
This commit is contained in:
parent
24bdc1de33
commit
dcbbcd23f5
19 changed files with 228 additions and 31 deletions
2
.github/workflows/ci.generate.ts
vendored
2
.github/workflows/ci.generate.ts
vendored
|
@ -791,7 +791,7 @@ const ci = {
|
|||
// Run unit then integration tests. Skip doc tests here
|
||||
// since they are sometimes very slow on Mac.
|
||||
"cargo test --locked --lib",
|
||||
"cargo test --locked --test '*'",
|
||||
"cargo test --locked --tests",
|
||||
].join("\n"),
|
||||
env: { CARGO_PROFILE_DEV_DEBUG: 0 },
|
||||
},
|
||||
|
|
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
|
@ -480,7 +480,7 @@ jobs:
|
|||
(startsWith(github.ref, 'refs/tags/') || matrix.os != 'linux'))
|
||||
run: |-
|
||||
cargo test --locked --lib
|
||||
cargo test --locked --test '*'
|
||||
cargo test --locked --tests
|
||||
env:
|
||||
CARGO_PROFILE_DEV_DEBUG: 0
|
||||
- name: Test (release)
|
||||
|
|
75
Cargo.lock
generated
75
Cargo.lock
generated
|
@ -607,6 +607,38 @@ version = "0.6.0"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "702fc72eb24e5a1e48ce58027a675bc24edd52096d5397d4aea7c6dd9eca0bd1"
|
||||
|
||||
[[package]]
|
||||
name = "cli_tests"
|
||||
version = "0.0.0"
|
||||
dependencies = [
|
||||
"bytes",
|
||||
"deno_ast",
|
||||
"deno_bench_util",
|
||||
"deno_core",
|
||||
"deno_fetch",
|
||||
"deno_lockfile",
|
||||
"deno_tls",
|
||||
"fastwebsockets",
|
||||
"flaky_test",
|
||||
"http 1.0.0",
|
||||
"http-body-util",
|
||||
"hyper 1.1.0",
|
||||
"hyper-util",
|
||||
"nix 0.26.2",
|
||||
"once_cell",
|
||||
"os_pipe",
|
||||
"pretty_assertions",
|
||||
"serde",
|
||||
"serde_repr",
|
||||
"test_util",
|
||||
"tokio",
|
||||
"tokio-util",
|
||||
"tower-lsp",
|
||||
"trust-dns-client",
|
||||
"trust-dns-server",
|
||||
"url",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "clipboard-win"
|
||||
version = "5.0.0"
|
||||
|
@ -1005,18 +1037,12 @@ dependencies = [
|
|||
"env_logger",
|
||||
"eszip",
|
||||
"fancy-regex",
|
||||
"fastwebsockets",
|
||||
"flaky_test",
|
||||
"flate2",
|
||||
"fs3",
|
||||
"fwdansi",
|
||||
"glibc_version",
|
||||
"glob",
|
||||
"hex",
|
||||
"http 1.0.0",
|
||||
"http-body-util",
|
||||
"hyper 1.1.0",
|
||||
"hyper-util",
|
||||
"import_map",
|
||||
"indexmap",
|
||||
"jsonc-parser",
|
||||
|
@ -1057,8 +1083,6 @@ dependencies = [
|
|||
"tokio",
|
||||
"tokio-util",
|
||||
"tower-lsp",
|
||||
"trust-dns-client",
|
||||
"trust-dns-server",
|
||||
"twox-hash",
|
||||
"typed-arena",
|
||||
"unicode-width",
|
||||
|
@ -1776,6 +1800,7 @@ dependencies = [
|
|||
"once_cell",
|
||||
"rustls",
|
||||
"rustls-pemfile",
|
||||
"rustls-tokio-stream",
|
||||
"rustls-webpki",
|
||||
"serde",
|
||||
"webpki-roots",
|
||||
|
@ -2838,7 +2863,7 @@ dependencies = [
|
|||
"presser",
|
||||
"thiserror",
|
||||
"winapi",
|
||||
"windows",
|
||||
"windows 0.51.1",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
|
@ -6309,6 +6334,7 @@ dependencies = [
|
|||
"termcolor",
|
||||
"tokio",
|
||||
"url",
|
||||
"win32job",
|
||||
"winapi",
|
||||
]
|
||||
|
||||
|
@ -7178,6 +7204,16 @@ version = "1.0.2"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "653f141f39ec16bba3c5abe400a0c60da7468261cc2cbf36805022876bc721a8"
|
||||
|
||||
[[package]]
|
||||
name = "win32job"
|
||||
version = "2.0.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "5b2b1bf557d947847a30eb73f79aa6cdb3eaf3ce02f5e9599438f77896a62b3c"
|
||||
dependencies = [
|
||||
"thiserror",
|
||||
"windows 0.52.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "winapi"
|
||||
version = "0.3.9"
|
||||
|
@ -7215,10 +7251,20 @@ version = "0.51.1"
|
|||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "ca229916c5ee38c2f2bc1e9d8f04df975b4bd93f9955dc69fabb5d91270045c9"
|
||||
dependencies = [
|
||||
"windows-core",
|
||||
"windows-core 0.51.1",
|
||||
"windows-targets 0.48.5",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "e48a53791691ab099e5e2ad123536d0fff50652600abaf43bbf952894110d0be"
|
||||
dependencies = [
|
||||
"windows-core 0.52.0",
|
||||
"windows-targets 0.52.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-core"
|
||||
version = "0.51.1"
|
||||
|
@ -7228,6 +7274,15 @@ dependencies = [
|
|||
"windows-targets 0.48.5",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-core"
|
||||
version = "0.52.0"
|
||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||
checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9"
|
||||
dependencies = [
|
||||
"windows-targets 0.52.0",
|
||||
]
|
||||
|
||||
[[package]]
|
||||
name = "windows-sys"
|
||||
version = "0.48.0"
|
||||
|
|
|
@ -6,6 +6,7 @@ members = [
|
|||
"bench_util",
|
||||
"cli",
|
||||
"cli/napi/sym",
|
||||
"cli/tests",
|
||||
"ext/broadcast_channel",
|
||||
"ext/cache",
|
||||
"ext/canvas",
|
||||
|
|
|
@ -4,6 +4,7 @@
|
|||
name = "deno"
|
||||
version = "1.40.4"
|
||||
authors.workspace = true
|
||||
autotests = false
|
||||
default-run = "deno"
|
||||
edition.workspace = true
|
||||
exclude = ["tests/testdata/npm/registry/*"]
|
||||
|
@ -16,6 +17,11 @@ name = "deno"
|
|||
path = "main.rs"
|
||||
doc = false
|
||||
|
||||
[[test]]
|
||||
name = "integration"
|
||||
path = "tests/integration_tests_runner.rs"
|
||||
harness = false
|
||||
|
||||
[[bench]]
|
||||
name = "deno_bench"
|
||||
harness = false
|
||||
|
@ -149,19 +155,8 @@ nix.workspace = true
|
|||
|
||||
[dev-dependencies]
|
||||
deno_bench_util.workspace = true
|
||||
deno_core = { workspace = true, features = ["include_js_files_for_snapshotting", "unsafe_use_unprotected_platform"] }
|
||||
fastwebsockets = { workspace = true, features = ["upgrade", "unstable-split"] }
|
||||
flaky_test = "=0.1.0"
|
||||
http.workspace = true
|
||||
http-body-util.workspace = true
|
||||
hyper.workspace = true
|
||||
hyper-util.workspace = true
|
||||
once_cell.workspace = true
|
||||
os_pipe.workspace = true
|
||||
pretty_assertions.workspace = true
|
||||
test_util.workspace = true
|
||||
trust-dns-client = "=0.22.0"
|
||||
trust-dns-server = "=0.22.1"
|
||||
|
||||
[package.metadata.winres]
|
||||
# This section defines the metadata that appears in the deno.exe PE header.
|
||||
|
|
51
cli/tests/Cargo.toml
Normal file
51
cli/tests/Cargo.toml
Normal file
|
@ -0,0 +1,51 @@
|
|||
# Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||
|
||||
[package]
|
||||
name = "cli_tests"
|
||||
version = "0.0.0"
|
||||
authors.workspace = true
|
||||
autotests = false
|
||||
edition.workspace = true
|
||||
license.workspace = true
|
||||
repository.workspace = true
|
||||
|
||||
[lib]
|
||||
path = "lib.rs"
|
||||
|
||||
[features]
|
||||
run = []
|
||||
|
||||
[[test]]
|
||||
name = "integration_tests"
|
||||
path = "integration_tests.rs"
|
||||
required-features = ["run"]
|
||||
|
||||
[dev-dependencies]
|
||||
bytes.workspace = true
|
||||
deno_ast.workspace = true
|
||||
deno_bench_util.workspace = true
|
||||
deno_core = { workspace = true, features = ["include_js_files_for_snapshotting", "unsafe_use_unprotected_platform"] }
|
||||
deno_fetch.workspace = true
|
||||
deno_lockfile.workspace = true
|
||||
deno_tls.workspace = true
|
||||
fastwebsockets = { workspace = true, features = ["upgrade", "unstable-split"] }
|
||||
flaky_test = "=0.1.0"
|
||||
http.workspace = true
|
||||
http-body-util.workspace = true
|
||||
hyper.workspace = true
|
||||
hyper-util.workspace = true
|
||||
once_cell.workspace = true
|
||||
os_pipe.workspace = true
|
||||
pretty_assertions.workspace = true
|
||||
serde.workspace = true
|
||||
serde_repr.workspace = true
|
||||
test_util.workspace = true
|
||||
tokio.workspace = true
|
||||
tokio-util.workspace = true
|
||||
tower-lsp.workspace = true
|
||||
trust-dns-client = "=0.22.0"
|
||||
trust-dns-server = "=0.22.1"
|
||||
url.workspace = true
|
||||
|
||||
[target.'cfg(unix)'.dev-dependencies]
|
||||
nix.workspace = true
|
|
@ -1,14 +1,14 @@
|
|||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||
|
||||
use deno_runtime::deno_net::ops_tls::TlsStream;
|
||||
use deno_runtime::deno_tls::rustls;
|
||||
use deno_runtime::deno_tls::rustls_pemfile;
|
||||
use lsp_types::Url;
|
||||
use deno_tls::rustls;
|
||||
use deno_tls::rustls_pemfile;
|
||||
use deno_tls::rustls_tokio_stream::TlsStream;
|
||||
use std::io::BufReader;
|
||||
use std::io::Cursor;
|
||||
use std::io::Read;
|
||||
use std::sync::Arc;
|
||||
use test_util as util;
|
||||
use url::Url;
|
||||
use util::testdata_path;
|
||||
use util::TestContext;
|
||||
|
||||
|
|
|
@ -6,7 +6,7 @@ use deno_core::error::AnyError;
|
|||
use deno_core::serde_json;
|
||||
use deno_core::serde_json::json;
|
||||
use deno_core::url;
|
||||
use deno_runtime::deno_fetch::reqwest;
|
||||
use deno_fetch::reqwest;
|
||||
use fastwebsockets::FragmentCollector;
|
||||
use fastwebsockets::Frame;
|
||||
use fastwebsockets::WebSocket;
|
||||
|
|
|
@ -7,7 +7,7 @@ use test_util as util;
|
|||
|
||||
util::unit_test_factory!(
|
||||
js_unit_test,
|
||||
"tests/unit",
|
||||
"../tests/unit",
|
||||
"*.ts",
|
||||
[
|
||||
abort_controller_test,
|
||||
|
|
|
@ -8,7 +8,7 @@ use util::env_vars_for_npm_tests;
|
|||
|
||||
util::unit_test_factory!(
|
||||
node_unit_test,
|
||||
"tests/unit_node",
|
||||
"../tests/unit_node",
|
||||
"**/*_test.ts",
|
||||
[
|
||||
_fs_access_test = _fs / _fs_access_test,
|
||||
|
|
|
@ -3,7 +3,7 @@
|
|||
use bytes::Bytes;
|
||||
use deno_core::serde_json::json;
|
||||
use deno_core::url;
|
||||
use deno_runtime::deno_fetch::reqwest;
|
||||
use deno_fetch::reqwest;
|
||||
use pretty_assertions::assert_eq;
|
||||
use std::io::Read;
|
||||
use std::io::Write;
|
||||
|
|
|
@ -3,4 +3,5 @@
|
|||
// The tests exist in a sub folder instead of as separate files in
|
||||
// this directory so that cargo doesn't compile each file as a new crate.
|
||||
|
||||
#[cfg(test)]
|
||||
mod integration;
|
||||
|
|
18
cli/tests/integration_tests_runner.rs
Normal file
18
cli/tests/integration_tests_runner.rs
Normal file
|
@ -0,0 +1,18 @@
|
|||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||
pub fn main() {
|
||||
let mut args = vec!["cargo", "test", "-p", "cli_tests", "--features", "run"];
|
||||
|
||||
if !cfg!(debug_assertions) {
|
||||
args.push("--release");
|
||||
}
|
||||
|
||||
args.push("--");
|
||||
|
||||
// If any args were passed to this process, pass them through to the child
|
||||
let orig_args = std::env::args().skip(1).collect::<Vec<_>>();
|
||||
let orig_args: Vec<&str> =
|
||||
orig_args.iter().map(|x| x.as_ref()).collect::<Vec<_>>();
|
||||
args.extend(orig_args);
|
||||
|
||||
test_util::spawn::exec_replace("cargo", &args).unwrap();
|
||||
}
|
1
cli/tests/lib.rs
Normal file
1
cli/tests/lib.rs
Normal file
|
@ -0,0 +1 @@
|
|||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
@ -19,6 +19,7 @@ deno_native_certs = "0.2.0"
|
|||
once_cell.workspace = true
|
||||
rustls = { workspace = true, features = ["dangerous_configuration"] }
|
||||
rustls-pemfile.workspace = true
|
||||
rustls-tokio-stream.workspace = true
|
||||
rustls-webpki.workspace = true
|
||||
serde.workspace = true
|
||||
webpki-roots.workspace = true
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
pub use deno_native_certs;
|
||||
pub use rustls;
|
||||
pub use rustls_pemfile;
|
||||
pub use rustls_tokio_stream;
|
||||
pub use webpki;
|
||||
pub use webpki_roots;
|
||||
|
||||
|
|
|
@ -53,6 +53,7 @@ tempfile.workspace = true
|
|||
termcolor.workspace = true
|
||||
tokio.workspace = true
|
||||
url.workspace = true
|
||||
win32job = "2"
|
||||
|
||||
[target.'cfg(windows)'.dependencies]
|
||||
winapi = { workspace = true, features = ["consoleapi", "synchapi", "handleapi", "namedpipeapi", "winbase", "winerror"] }
|
||||
|
|
|
@ -32,6 +32,7 @@ pub mod lsp;
|
|||
mod npm;
|
||||
pub mod pty;
|
||||
pub mod servers;
|
||||
pub mod spawn;
|
||||
|
||||
pub use builders::DenoChild;
|
||||
pub use builders::TestCommandBuilder;
|
||||
|
|
71
test_util/src/spawn.rs
Normal file
71
test_util/src/spawn.rs
Normal file
|
@ -0,0 +1,71 @@
|
|||
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
||||
use anyhow::Error;
|
||||
use std::convert::Infallible;
|
||||
|
||||
/// For unix targets, we just replace our current process with the desired cargo process.
|
||||
#[cfg(unix)]
|
||||
pub fn exec_replace_inner(
|
||||
cmd: &str,
|
||||
args: &[&str],
|
||||
) -> Result<Infallible, Error> {
|
||||
use std::ffi::CStr;
|
||||
use std::ffi::CString;
|
||||
|
||||
let args = args
|
||||
.iter()
|
||||
.map(|arg| CString::new(*arg).unwrap())
|
||||
.collect::<Vec<_>>();
|
||||
let args: Vec<&CStr> =
|
||||
args.iter().map(|arg| arg.as_ref()).collect::<Vec<_>>();
|
||||
|
||||
let err = nix::unistd::execvp(&CString::new(cmd).unwrap(), &args)
|
||||
.expect_err("Impossible");
|
||||
Err(err.into())
|
||||
}
|
||||
|
||||
#[cfg(windows)]
|
||||
pub fn exec_replace_inner(
|
||||
cmd: &str,
|
||||
args: &[&str],
|
||||
) -> Result<Infallible, Error> {
|
||||
use std::os::windows::io::AsRawHandle;
|
||||
use std::process::Command;
|
||||
use win32job::ExtendedLimitInfo;
|
||||
use win32job::Job;
|
||||
|
||||
// Use a job to ensure the child process's lifetime does not exceed the current process's lifetime.
|
||||
// This ensures that if the current process is terminated (e.g., via ctrl+c or task manager),
|
||||
// the child process is automatically reaped.
|
||||
|
||||
// For more information about this technique, see Raymond Chen's blog post:
|
||||
// https://devblogs.microsoft.com/oldnewthing/20131209-00/?p=2433
|
||||
// Note: While our implementation is not perfect, it serves its purpose for test code.
|
||||
|
||||
// In the future, we may directly obtain the main thread's handle from Rust code and use it
|
||||
// to create a suspended process that we can then resume:
|
||||
// https://github.com/rust-lang/rust/issues/96723
|
||||
|
||||
// Creates a child process and assigns it to our current job.
|
||||
// A more reliable approach would be to create the child suspended and then assign it to the job.
|
||||
// For now, we create the child, create the job, and then assign both us and the child to the job.
|
||||
let mut child = Command::new(cmd).args(&args[1..]).spawn()?;
|
||||
|
||||
let mut info = ExtendedLimitInfo::default();
|
||||
info.limit_kill_on_job_close();
|
||||
let job = Job::create_with_limit_info(&info)?;
|
||||
job.assign_current_process()?;
|
||||
let handle = child.as_raw_handle();
|
||||
job.assign_process(handle as _)?;
|
||||
|
||||
let exit = child.wait()?;
|
||||
std::process::exit(exit.code().unwrap_or(1));
|
||||
}
|
||||
|
||||
/// Runs a command, replacing the current process on Unix. On Windows, this function blocks and
|
||||
/// exits.
|
||||
///
|
||||
/// In either case, the only way this function returns is if it fails to launch the child
|
||||
/// process.
|
||||
pub fn exec_replace(command: &str, args: &[&str]) -> Result<Infallible, Error> {
|
||||
exec_replace_inner(command, args)
|
||||
}
|
Loading…
Reference in a new issue