Description
The fingerprint of a build script is not included for dependencies. #6720 added this tracking, but because it requires mtimes to work correctly, #6734 removed it for registry dependencies. This is because dependencies are often cached in Docker images, and Docker zeros out the nanoseconds portion of the mtime when the image is saved.
This only affects the scenario when two separate commands are run, such as cargo build
then cargo test
. The first command will pick up the changes, but the second will not.
There are a few ways that I can think of that this would occur: changing an environment variable tracked by rerun-if-env-changed
, a system library changes that is tracked by rerun-if-changed
, or adding a build override after already building.
I cannot think of an easy way to fix this. I'd rather not try to change the mtime resolution in the fingerprint. I think the ideal solution would be: instead of tracking the invoked.timestamp
, include the same fingerprint information calculated by build_script_local_fingerprints
when computing dependencies, and use a hybrid mtime/content hash for rerun-if-changed
files.
Below are some tests that demonstrate this behavior. Both of these will fail on the last step if #6734 is merged.
#[test]
fn rerun_build_script_dep() {
Package::new("bar", "0.1.0")
.file(
"build.rs",
r#"
fn main() {
println!("cargo:rerun-if-env-changed=MY_TEST_VAR");
}
"#,
)
.file("src/lib.rs", "")
.publish();
let p = project()
.file(
"Cargo.toml",
r#"
[package]
name = "foo"
version = "0.1.0"
[dependencies]
bar = "0.1"
"#,
)
.file("src/lib.rs", "")
.build();
p.cargo("build").env("MY_TEST_VAR", "1").run();
p.cargo("test --lib").env("MY_TEST_VAR", "1").run();
p.cargo("build -vv")
.env("MY_TEST_VAR", "1")
.with_stderr_contains("[FRESH] bar v0.1.0")
.with_stderr_contains("[FRESH] foo v0.1.0 [..]")
.run();
p.cargo("build -vv")
.env("MY_TEST_VAR", "2")
.with_stderr_contains("[COMPILING] bar v0.1.0")
.with_stderr_contains("[COMPILING] foo v0.1.0 [..]")
.run();
p.cargo("test -vv --lib")
.env("MY_TEST_VAR", "2")
.with_stderr_contains("[FRESH] bar v0.1.0")
.with_stderr_contains("[COMPILING] foo v0.1.0 [..]")
.run();
}
#[test]
fn changing_dep_override_invalidates() {
let target = rustc_host();
Package::new("dep1", "0.1.0")
.file(
"Cargo.toml",
r#"
[project]
name = "dep1"
version = "0.1.0"
links = "asdf"
"#,
)
.file("build.rs", "fn main() {}")
.file("src/lib.rs", "pub fn f() -> bool { cfg!(trigger) }")
.publish();
let p = project()
.file(
"Cargo.toml",
r#"
[project]
name = "foo"
version = "0.5.0"
[dependencies]
dep1 = "0.1"
"#,
)
.file(
"src/lib.rs",
r#"
extern crate dep1;
#[test]
fn t1() {
assert_eq!(dep1::f(), true, "trigger not set");
}
"#,
)
.build();
p.cargo("build").run();
p.cargo("test --lib -vv")
.with_stderr_contains("[FRESH] dep1 v0.1.0")
.with_stdout_contains("[..]trigger not set[..]")
.with_status(101)
.run();
p.change_file(
".cargo/config",
&format!(
r#"
[target.{}.asdf]
rustc-cfg = ["trigger"]
"#,
target
),
);
p.cargo("build -vv")
.with_stderr_contains("[COMPILING] dep1 v0.1.0")
.with_stderr_contains("[COMPILING] foo [..]")
.run();
p.cargo("test --lib -vv")
.with_stderr_contains("[FRESH] dep1")
.with_stderr_contains("[COMPILING] foo [..]")
.run();
}