Skip to content

Commit eca9e15

Browse files
authored
Auto merge of #3242 - jhbabon:cfgs-flags-per-package, r=alexcrichton
FIX: Call rustdoc test with the correct cfg flags of a package. There was a situation in which if you you had a lib that depends on a package with features, whenever you ran the tests for the package the `rustdoc test` call was failing because it was called with the root `cfg` flags, not the package `cfg` flags. This fix solves the issue by keeping track of the `cfg` flags per package, so the `rustdoc` command will be generated with the correct `cfg` flags. Closes #3172
2 parents 77e36ed + 665aa7f commit eca9e15

File tree

5 files changed

+89
-18
lines changed

5 files changed

+89
-18
lines changed

src/cargo/ops/cargo_rustc/compilation.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,8 @@ pub struct Compilation<'cfg> {
4343

4444
pub to_doc_test: Vec<Package>,
4545

46-
/// Features enabled during this compilation.
47-
pub cfgs: HashSet<String>,
46+
/// Features per package enabled during this compilation.
47+
pub cfgs: HashMap<PackageId, HashSet<String>>,
4848

4949
pub target: String,
5050

@@ -63,7 +63,7 @@ impl<'cfg> Compilation<'cfg> {
6363
binaries: Vec::new(),
6464
extra_env: HashMap::new(),
6565
to_doc_test: Vec::new(),
66-
cfgs: HashSet::new(),
66+
cfgs: HashMap::new(),
6767
config: config,
6868
target: String::new(),
6969
}

src/cargo/ops/cargo_rustc/mod.rs

+10-11
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use std::collections::HashMap;
1+
use std::collections::{HashMap, HashSet};
22
use std::env;
33
use std::ffi::{OsStr, OsString};
44
use std::fs;
@@ -81,7 +81,6 @@ pub fn compile_targets<'a, 'cfg: 'a>(ws: &Workspace<'cfg>,
8181
})
8282
}).collect::<Vec<_>>();
8383

84-
let root = try!(ws.current());
8584
let mut cx = try!(Context::new(ws, resolve, packages, config,
8685
build_config, profiles));
8786

@@ -142,19 +141,19 @@ pub fn compile_targets<'a, 'cfg: 'a>(ws: &Workspace<'cfg>,
142141
cx.compilation.libraries.insert(pkgid.clone(), v);
143142
}
144143
}
145-
}
146144

147-
let root_pkg = root.package_id();
148-
if let Some(feats) = cx.resolve.features(root_pkg) {
149-
cx.compilation.cfgs.extend(feats.iter().map(|feat| {
150-
format!("feature=\"{}\"", feat)
151-
}));
145+
if let Some(feats) = cx.resolve.features(&unit.pkg.package_id()) {
146+
cx.compilation.cfgs.entry(unit.pkg.package_id().clone())
147+
.or_insert(HashSet::new())
148+
.extend(feats.iter().map(|feat| format!("feature=\"{}\"", feat)));
149+
}
152150
}
153151

154152
for (&(ref pkg, _), output) in cx.build_state.outputs.lock().unwrap().iter() {
155-
if pkg == root_pkg {
156-
cx.compilation.cfgs.extend(output.cfgs.iter().cloned());
157-
}
153+
cx.compilation.cfgs.entry(pkg.clone())
154+
.or_insert(HashSet::new())
155+
.extend(output.cfgs.iter().cloned());
156+
158157
for dir in output.library_paths.iter() {
159158
cx.compilation.native_dirs.insert(dir.clone());
160159
}

src/cargo/ops/cargo_test.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -145,8 +145,10 @@ fn run_doc_tests(options: &TestOptions,
145145
p.arg("--test-args").arg(&test_args.join(" "));
146146
}
147147

148-
for cfg in compilation.cfgs.iter() {
149-
p.arg("--cfg").arg(cfg);
148+
if let Some(cfgs) = compilation.cfgs.get(&package.package_id()) {
149+
for cfg in cfgs.iter() {
150+
p.arg("--cfg").arg(cfg);
151+
}
150152
}
151153

152154
for (_, libs) in compilation.libraries.iter() {

src/doc/machine-readable-output.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ Cargo can output information about project and build in JSON format.
77
You can use `cargo metadata` command to get information about project structure
88
and dependencies. The output of the command looks like this:
99

10-
```
10+
```text
1111
{
1212
// Integer version number of the format.
1313
"version": integer,
@@ -59,7 +59,7 @@ without waiting for the whole build to finish.
5959

6060
The message format looks like this:
6161

62-
```
62+
```text
6363
{
6464
// Type of the message.
6565
"reason": "compiler-message",

tests/test.rs

+70
Original file line numberDiff line numberDiff line change
@@ -2262,3 +2262,73 @@ fn panic_abort_multiple() {
22622262
.arg("-p").arg("a"),
22632263
execs().with_status(0));
22642264
}
2265+
2266+
#[test]
2267+
fn pass_correct_cfgs_flags_to_rustdoc() {
2268+
let p = project("foo")
2269+
.file("Cargo.toml", r#"
2270+
[package]
2271+
name = "foo"
2272+
version = "0.1.0"
2273+
authors = []
2274+
2275+
[features]
2276+
default = ["feature_a/default"]
2277+
nightly = ["feature_a/nightly"]
2278+
2279+
[dependencies.feature_a]
2280+
path = "libs/feature_a"
2281+
default-features = false
2282+
"#)
2283+
.file("src/lib.rs", r#"
2284+
#[cfg(test)]
2285+
mod tests {
2286+
#[test]
2287+
fn it_works() {
2288+
assert!(true);
2289+
}
2290+
}
2291+
"#)
2292+
.file("libs/feature_a/Cargo.toml", r#"
2293+
[package]
2294+
name = "feature_a"
2295+
version = "0.1.0"
2296+
authors = []
2297+
2298+
[features]
2299+
default = ["serde_codegen"]
2300+
nightly = ["serde_derive"]
2301+
2302+
[dependencies]
2303+
serde_derive = { version = "0.8", optional = true }
2304+
2305+
[build-dependencies]
2306+
serde_codegen = { version = "0.8", optional = true }
2307+
"#)
2308+
.file("libs/feature_a/src/lib.rs", r#"
2309+
#[cfg(feature = "serde_derive")]
2310+
const MSG: &'static str = "This is safe";
2311+
2312+
#[cfg(feature = "serde_codegen")]
2313+
const MSG: &'static str = "This is risky";
2314+
2315+
pub fn get() -> &'static str {
2316+
MSG
2317+
}
2318+
"#);
2319+
2320+
assert_that(p.cargo_process("test")
2321+
.arg("--package").arg("feature_a")
2322+
.arg("--verbose"),
2323+
execs().with_status(0)
2324+
.with_stderr_contains("\
2325+
[DOCTEST] feature_a
2326+
[RUNNING] `rustdoc --test [..]serde_codegen[..]`"));
2327+
2328+
assert_that(p.cargo_process("test")
2329+
.arg("--verbose"),
2330+
execs().with_status(0)
2331+
.with_stderr_contains("\
2332+
[DOCTEST] foo
2333+
[RUNNING] `rustdoc --test [..]feature_a[..]`"));
2334+
}

0 commit comments

Comments
 (0)