Skip to content

Commit 3aaa8f9

Browse files
committed
Auto merge of #10172 - ehuss:doc-lib-before-bin, r=alexcrichton
Document lib before bin. This changes it so that documenting a library is required to finish before documenting a binary. The issue is that the binary may have intra-doc links to the library. If they are documented concurrently, then the links will sometimes fail (since it is a race). Or, if doing `cargo doc --bins`, then the library docs wouldn't exist at all. Note that in the tests this introduces some more name collisions if you just run `cargo doc --bins` and there is a colliding library/binary name. There is some risk that someone might be trying to run the commands separately to get around the collision error, but I think it is unlikely.
2 parents fe69af3 + a09fbf2 commit 3aaa8f9

File tree

2 files changed

+76
-7
lines changed

2 files changed

+76
-7
lines changed

src/cargo/core/compiler/unit_dependencies.rs

+7-3
Original file line numberDiff line numberDiff line change
@@ -326,7 +326,7 @@ fn compute_deps(
326326
if unit.target.is_lib() && unit.mode != CompileMode::Doctest {
327327
return Ok(ret);
328328
}
329-
ret.extend(maybe_lib(unit, state, unit_for)?);
329+
ret.extend(maybe_lib(unit, state, unit_for, None)?);
330330

331331
// If any integration tests/benches are being run, make sure that
332332
// binaries are built as well.
@@ -469,7 +469,10 @@ fn compute_deps_doc(
469469

470470
// If we document a binary/example, we need the library available.
471471
if unit.target.is_bin() || unit.target.is_example() {
472-
ret.extend(maybe_lib(unit, state, unit_for)?);
472+
// build the lib
473+
ret.extend(maybe_lib(unit, state, unit_for, None)?);
474+
// and also the lib docs for intra-doc links
475+
ret.extend(maybe_lib(unit, state, unit_for, Some(unit.mode))?);
473476
}
474477

475478
// Add all units being scraped for examples as a dependency of Doc units.
@@ -497,13 +500,14 @@ fn maybe_lib(
497500
unit: &Unit,
498501
state: &mut State<'_, '_>,
499502
unit_for: UnitFor,
503+
force_mode: Option<CompileMode>,
500504
) -> CargoResult<Option<UnitDep>> {
501505
unit.pkg
502506
.targets()
503507
.iter()
504508
.find(|t| t.is_linkable())
505509
.map(|t| {
506-
let mode = check_or_build_mode(unit.mode, t);
510+
let mode = force_mode.unwrap_or_else(|| check_or_build_mode(unit.mode, t));
507511
let dep_unit_for = unit_for.with_dependency(unit, t);
508512
new_unit_dep(
509513
state,

tests/testsuite/doc.rs

+69-4
Original file line numberDiff line numberDiff line change
@@ -464,8 +464,17 @@ fn doc_lib_bin_same_name_documents_named_bin_when_requested() {
464464
.build();
465465

466466
p.cargo("doc --bin foo")
467-
.with_stderr(
467+
// The checking/documenting lines are sometimes swapped since they run
468+
// concurrently.
469+
.with_stderr_unordered(
468470
"\
471+
warning: output filename collision.
472+
The bin target `foo` in package `foo v0.0.1 ([ROOT]/foo)` \
473+
has the same output filename as the lib target `foo` in package `foo v0.0.1 ([ROOT]/foo)`.
474+
Colliding filename is: [ROOT]/foo/target/doc/foo/index.html
475+
The targets should have unique names.
476+
This is a known bug where multiple crates with the same name use
477+
the same path; see <https://github.com/rust-lang/cargo/issues/6313>.
469478
[CHECKING] foo v0.0.1 ([CWD])
470479
[DOCUMENTING] foo v0.0.1 ([CWD])
471480
[FINISHED] dev [unoptimized + debuginfo] target(s) in [..]
@@ -500,8 +509,17 @@ fn doc_lib_bin_same_name_documents_bins_when_requested() {
500509
.build();
501510

502511
p.cargo("doc --bins")
503-
.with_stderr(
512+
// The checking/documenting lines are sometimes swapped since they run
513+
// concurrently.
514+
.with_stderr_unordered(
504515
"\
516+
warning: output filename collision.
517+
The bin target `foo` in package `foo v0.0.1 ([ROOT]/foo)` \
518+
has the same output filename as the lib target `foo` in package `foo v0.0.1 ([ROOT]/foo)`.
519+
Colliding filename is: [ROOT]/foo/target/doc/foo/index.html
520+
The targets should have unique names.
521+
This is a known bug where multiple crates with the same name use
522+
the same path; see <https://github.com/rust-lang/cargo/issues/6313>.
505523
[CHECKING] foo v0.0.1 ([CWD])
506524
[DOCUMENTING] foo v0.0.1 ([CWD])
507525
[FINISHED] dev [unoptimized + debuginfo] target(s) in [..]
@@ -543,7 +561,9 @@ fn doc_lib_bin_example_same_name_documents_named_example_when_requested() {
543561
.build();
544562

545563
p.cargo("doc --example ex1")
546-
.with_stderr(
564+
// The checking/documenting lines are sometimes swapped since they run
565+
// concurrently.
566+
.with_stderr_unordered(
547567
"\
548568
[CHECKING] foo v0.0.1 ([CWD])
549569
[DOCUMENTING] foo v0.0.1 ([CWD])
@@ -594,7 +614,9 @@ fn doc_lib_bin_example_same_name_documents_examples_when_requested() {
594614
.build();
595615

596616
p.cargo("doc --examples")
597-
.with_stderr(
617+
// The checking/documenting lines are sometimes swapped since they run
618+
// concurrently.
619+
.with_stderr_unordered(
598620
"\
599621
[CHECKING] foo v0.0.1 ([CWD])
600622
[DOCUMENTING] foo v0.0.1 ([CWD])
@@ -2365,3 +2387,46 @@ fn scrape_examples_missing_flag() {
23652387
.with_stderr("error: -Z rustdoc-scrape-examples must take [..] an argument")
23662388
.run();
23672389
}
2390+
2391+
#[cargo_test]
2392+
fn lib_before_bin() {
2393+
// Checks that the library is documented before the binary.
2394+
// Previously they were built concurrently, which can cause issues
2395+
// if the bin has intra-doc links to the lib.
2396+
let p = project()
2397+
.file(
2398+
"src/lib.rs",
2399+
r#"
2400+
/// Hi
2401+
pub fn abc() {}
2402+
"#,
2403+
)
2404+
.file(
2405+
"src/bin/somebin.rs",
2406+
r#"
2407+
//! See [`foo::abc`]
2408+
fn main() {}
2409+
"#,
2410+
)
2411+
.build();
2412+
2413+
// Run check first. This just helps ensure that the test clearly shows the
2414+
// order of the rustdoc commands.
2415+
p.cargo("check").run();
2416+
2417+
// The order of output here should be deterministic.
2418+
p.cargo("doc -v")
2419+
.with_stderr(
2420+
"\
2421+
[DOCUMENTING] foo [..]
2422+
[RUNNING] `rustdoc --crate-type lib --crate-name foo src/lib.rs [..]
2423+
[RUNNING] `rustdoc --crate-type bin --crate-name somebin src/bin/somebin.rs [..]
2424+
[FINISHED] [..]
2425+
",
2426+
)
2427+
.run();
2428+
2429+
// And the link should exist.
2430+
let bin_html = p.read_file("target/doc/somebin/index.html");
2431+
assert!(bin_html.contains("../foo/fn.abc.html"));
2432+
}

0 commit comments

Comments
 (0)