Skip to content

Commit 6b94ed2

Browse files
committed
Infer targets from subdirectories
1 parent 525323f commit 6b94ed2

File tree

3 files changed

+125
-43
lines changed

3 files changed

+125
-43
lines changed

src/cargo/util/toml/targets.rs

+45-41
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
//! * `src/bin/*.rs` are binaries
66
//! * `examples/*.rs` are examples
77
//! * `tests/*.rs` are integration tests
8+
//! * `benches/*.rs` are benchmarks
89
//!
910
//! It is a bit tricky because we need match explicit information from `Cargo.toml`
1011
//! with implicit info in directory layout.
@@ -218,8 +219,11 @@ fn clean_examples(toml_examples: Option<&Vec<TomlExampleTarget>>,
218219
package_root: &Path,
219220
errors: &mut Vec<String>)
220221
-> CargoResult<Vec<Target>> {
222+
223+
let inferred = infer_from_directory(&package_root.join("examples"));
224+
221225
let targets = clean_targets("example", "example",
222-
toml_examples, inferred_examples(package_root),
226+
toml_examples, inferred,
223227
package_root, errors)?;
224228

225229
let mut result = Vec::new();
@@ -241,8 +245,11 @@ fn clean_examples(toml_examples: Option<&Vec<TomlExampleTarget>>,
241245
fn clean_tests(toml_tests: Option<&Vec<TomlTestTarget>>,
242246
package_root: &Path,
243247
errors: &mut Vec<String>) -> CargoResult<Vec<Target>> {
248+
249+
let inferred = infer_from_directory(&package_root.join("tests"));
250+
244251
let targets = clean_targets("test", "test",
245-
toml_tests, inferred_tests(package_root),
252+
toml_tests, inferred,
246253
package_root, errors)?;
247254

248255
let mut result = Vec::new();
@@ -272,8 +279,10 @@ fn clean_benches(toml_benches: Option<&Vec<TomlBenchTarget>>,
272279
Some(legacy_path)
273280
};
274281

282+
let inferred = infer_from_directory(&package_root.join("benches"));
283+
275284
let targets = clean_targets_with_legacy_path("benchmark", "bench",
276-
toml_benches, inferred_benches(package_root),
285+
toml_benches, inferred,
277286
package_root,
278287
errors,
279288
&mut legacy_bench_path)?;
@@ -359,41 +368,9 @@ fn inferred_bins(package_root: &Path, package_name: &str) -> Vec<(String, PathBu
359368
}
360369
result.extend(infer_from_directory(&package_root.join("src").join("bin")));
361370

362-
if let Ok(entries) = fs::read_dir(&package_root.join("src").join("bin")) {
363-
let multifile_bins = entries
364-
.filter_map(|e| e.ok())
365-
.filter(is_not_dotfile)
366-
.filter(|e| match e.file_type() {
367-
Ok(t) if t.is_dir() => true,
368-
_ => false
369-
})
370-
.filter_map(|entry| {
371-
let dir = entry.path();
372-
let main = dir.join("main.rs");
373-
let name = dir.file_name().and_then(|n| n.to_str());
374-
match (main.exists(), name) {
375-
(true, Some(name)) => Some((name.to_owned(), main)),
376-
_ => None
377-
}
378-
});
379-
result.extend(multifile_bins);
380-
}
381-
382371
result
383372
}
384373

385-
fn inferred_examples(package_root: &Path) -> Vec<(String, PathBuf)> {
386-
infer_from_directory(&package_root.join("examples"))
387-
}
388-
389-
fn inferred_tests(package_root: &Path) -> Vec<(String, PathBuf)> {
390-
infer_from_directory(&package_root.join("tests"))
391-
}
392-
393-
fn inferred_benches(package_root: &Path) -> Vec<(String, PathBuf)> {
394-
infer_from_directory(&package_root.join("benches"))
395-
}
396-
397374
fn infer_from_directory(directory: &Path) -> Vec<(String, PathBuf)> {
398375
let entries = match fs::read_dir(directory) {
399376
Err(_) => return Vec::new(),
@@ -403,15 +380,42 @@ fn infer_from_directory(directory: &Path) -> Vec<(String, PathBuf)> {
403380
entries
404381
.filter_map(|e| e.ok())
405382
.filter(is_not_dotfile)
406-
.map(|e| e.path())
407-
.filter(|f| f.extension().and_then(|s| s.to_str()) == Some("rs"))
408-
.filter_map(|f| {
409-
f.file_stem().and_then(|s| s.to_str())
410-
.map(|s| (s.to_owned(), f.clone()))
411-
})
383+
.filter_map(infer_any)
412384
.collect()
413385
}
414386

387+
388+
fn infer_any(entry: DirEntry) -> Option<(String, PathBuf)> {
389+
if entry.path().extension().and_then(|p| p.to_str()) == Some("rs") {
390+
infer_file(entry)
391+
} else if entry.file_type().map(|t| t.is_dir()).ok() == Some(true) {
392+
infer_subdirectory(entry)
393+
} else {
394+
None
395+
}
396+
}
397+
398+
399+
fn infer_file(entry: DirEntry) -> Option<(String, PathBuf)> {
400+
let path = entry.path();
401+
path
402+
.file_stem()
403+
.and_then(|p| p.to_str())
404+
.map(|p| (p.to_owned(), path.clone()))
405+
}
406+
407+
408+
fn infer_subdirectory(entry: DirEntry) -> Option<(String, PathBuf)> {
409+
let path = entry.path();
410+
let main = path.join("main.rs");
411+
let name = path.file_name().and_then(|n| n.to_str());
412+
match (name, main.exists()) {
413+
(Some(name), true) => Some((name.to_owned(), main)),
414+
_ => None
415+
}
416+
}
417+
418+
415419
fn is_not_dotfile(entry: &DirEntry) -> bool {
416420
entry.file_name().to_str().map(|s| s.starts_with('.')) == Some(false)
417421
}

src/doc/manifest.md

+8-1
Original file line numberDiff line numberDiff line change
@@ -532,7 +532,8 @@ each file you want to build.
532532

533533
Your project can optionally contain folders named `examples`, `tests`, and
534534
`benches`, which Cargo will treat as containing examples,
535-
integration tests, and benchmarks respectively.
535+
integration tests, and benchmarks respectively. Analogous to `bin` targets, they
536+
may be composed of single files or directories with a `main.rs` file.
536537

537538
```notrust
538539
▾ src/ # directory containing source files
@@ -544,10 +545,16 @@ integration tests, and benchmarks respectively.
544545
main.rs
545546
▾ examples/ # (optional) examples
546547
*.rs
548+
▾ */ # (optional) directories containing multi-file examples
549+
main.rs
547550
▾ tests/ # (optional) integration tests
548551
*.rs
552+
▾ */ # (optional) directories containing multi-file tests
553+
main.rs
549554
▾ benches/ # (optional) benchmarks
550555
*.rs
556+
▾ */ # (optional) directories containing multi-file benchmarks
557+
main.rs
551558
```
552559

553560
To structure your code after you've created the files and folders for your

tests/build.rs

+72-1
Original file line numberDiff line numberDiff line change
@@ -3495,7 +3495,7 @@ fn dir_and_file_with_same_name_in_bin() {
34953495
.file("src/bin/foo.rs", "fn main() {}")
34963496
.file("src/bin/foo/main.rs", "fn main() {}");
34973497

3498-
assert_that(p.cargo_process("build"),
3498+
assert_that(p.cargo_process("build"),
34993499
execs().with_status(101)
35003500
.with_stderr_contains("\
35013501
[..]found duplicate binary name foo, but all binary targets must have a unique name[..]
@@ -3521,6 +3521,77 @@ fn inferred_path_in_src_bin_foo() {
35213521
assert_that(&p.bin("bar"), existing_file());
35223522
}
35233523

3524+
#[test]
3525+
fn inferred_examples() {
3526+
let p = project("foo")
3527+
.file("Cargo.toml", r#"
3528+
[package]
3529+
name = "foo"
3530+
version = "0.1.0"
3531+
authors = []
3532+
"#)
3533+
.file("src/lib.rs", "fn main() {}")
3534+
.file("examples/bar.rs", "fn main() {}")
3535+
.file("examples/baz/main.rs", "fn main() {}");
3536+
3537+
assert_that(p.cargo_process("test"), execs().with_status(0));
3538+
assert_that(&p.bin("examples/bar"), existing_file());
3539+
assert_that(&p.bin("examples/baz"), existing_file());
3540+
}
3541+
3542+
#[test]
3543+
fn inferred_tests() {
3544+
let p = project("foo")
3545+
.file("Cargo.toml", r#"
3546+
[package]
3547+
name = "foo"
3548+
version = "0.1.0"
3549+
authors = []
3550+
"#)
3551+
.file("src/lib.rs", "fn main() {}")
3552+
.file("tests/bar.rs", "fn main() {}")
3553+
.file("tests/baz/main.rs", "fn main() {}");
3554+
3555+
assert_that(
3556+
p.cargo_process("test").arg("--test=bar").arg("--test=baz"),
3557+
execs().with_status(0));
3558+
}
3559+
3560+
#[test]
3561+
fn inferred_benchmark() {
3562+
let p = project("foo")
3563+
.file("Cargo.toml", r#"
3564+
[package]
3565+
name = "foo"
3566+
version = "0.1.0"
3567+
authors = []
3568+
"#)
3569+
.file("src/lib.rs", "fn main() {}")
3570+
.file("benches/bar.rs", "fn main() {}");
3571+
3572+
assert_that(
3573+
p.cargo_process("bench").arg("--bench=bar"),
3574+
execs().with_status(0));
3575+
}
3576+
3577+
#[test]
3578+
fn inferred_benchmark_from_directory() {
3579+
//FIXME: merge with `inferred_benchmark` after fixing #4504
3580+
let p = project("foo")
3581+
.file("Cargo.toml", r#"
3582+
[package]
3583+
name = "foo"
3584+
version = "0.1.0"
3585+
authors = []
3586+
"#)
3587+
.file("src/lib.rs", "fn main() {}")
3588+
.file("benches/bar/main.rs", "fn main() {}");
3589+
3590+
assert_that(
3591+
p.cargo_process("bench").arg("--bench=bar"),
3592+
execs().with_status(0));
3593+
}
3594+
35243595
#[test]
35253596
fn same_metadata_different_directory() {
35263597
// A top-level crate built in two different workspaces should have the

0 commit comments

Comments
 (0)