Skip to content

Commit 95d16be

Browse files
committed
Auto merge of #13960 - tweag:issue-13695, r=weihanglo
Include vcs_info even if workspace is dirty ### What does this PR try to resolve? Related to #13695. Generates and packages the `.cargo_vcs_info.json` file even if the worktree is dirty, as long as `--allow-dirty` is passed. Also added a `dirty` field to the file to record if the Git repository status is dirty. Tests are included.
2 parents e18afe2 + e7bfed1 commit 95d16be

File tree

8 files changed

+162
-26
lines changed

8 files changed

+162
-26
lines changed

src/cargo/ops/cargo_package.rs

+20-14
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,9 @@ struct VcsInfo {
8181
#[derive(Serialize)]
8282
struct GitVcsInfo {
8383
sha1: String,
84+
/// Indicate whether or not the Git worktree is dirty.
85+
#[serde(skip_serializing_if = "std::ops::Not::not")]
86+
dirty: bool,
8487
}
8588

8689
/// Packages a single package in a workspace, returning the resulting tar file.
@@ -235,14 +238,8 @@ fn prepare_archive(
235238
}
236239
let src_files = src.list_files(pkg)?;
237240

238-
// Check (git) repository state, getting the current commit hash if not
239-
// dirty.
240-
let vcs_info = if !opts.allow_dirty {
241-
// This will error if a dirty repo is found.
242-
check_repo_state(pkg, &src_files, gctx)?
243-
} else {
244-
None
245-
};
241+
// Check (git) repository state, getting the current commit hash.
242+
let vcs_info = check_repo_state(pkg, &src_files, gctx, &opts)?;
246243

247244
build_ar_list(ws, pkg, src_files, vcs_info)
248245
}
@@ -559,13 +556,15 @@ fn check_metadata(pkg: &Package, gctx: &GlobalContext) -> CargoResult<()> {
559556
}
560557

561558
/// Checks if the package source is in a *git* DVCS repository. If *git*, and
562-
/// the source is *dirty* (e.g., has uncommitted changes) then `bail!` with an
563-
/// informative message. Otherwise return the sha1 hash of the current *HEAD*
564-
/// commit, or `None` if no repo is found.
559+
/// the source is *dirty* (e.g., has uncommitted changes), and `--allow-dirty`
560+
/// has not been passed, then `bail!` with an informative message. Otherwise
561+
/// return the sha1 hash of the current *HEAD* commit, or `None` if no repo is
562+
/// found.
565563
fn check_repo_state(
566564
p: &Package,
567565
src_files: &[PathBuf],
568566
gctx: &GlobalContext,
567+
opts: &PackageOpts<'_>,
569568
) -> CargoResult<Option<VcsInfo>> {
570569
if let Ok(repo) = git2::Repository::discover(p.root()) {
571570
if let Some(workdir) = repo.workdir() {
@@ -585,7 +584,7 @@ fn check_repo_state(
585584
.unwrap_or("")
586585
.replace("\\", "/");
587586
return Ok(Some(VcsInfo {
588-
git: git(p, src_files, &repo)?,
587+
git: git(p, src_files, &repo, &opts)?,
589588
path_in_vcs,
590589
}));
591590
}
@@ -608,7 +607,12 @@ fn check_repo_state(
608607
// directory is dirty or not, thus we have to assume that it's clean.
609608
return Ok(None);
610609

611-
fn git(p: &Package, src_files: &[PathBuf], repo: &git2::Repository) -> CargoResult<GitVcsInfo> {
610+
fn git(
611+
p: &Package,
612+
src_files: &[PathBuf],
613+
repo: &git2::Repository,
614+
opts: &PackageOpts<'_>,
615+
) -> CargoResult<GitVcsInfo> {
612616
// This is a collection of any dirty or untracked files. This covers:
613617
// - new/modified/deleted/renamed/type change (index or worktree)
614618
// - untracked files (which are "new" worktree files)
@@ -633,10 +637,12 @@ fn check_repo_state(
633637
.to_string()
634638
})
635639
.collect();
636-
if dirty_src_files.is_empty() {
640+
let dirty = !dirty_src_files.is_empty();
641+
if !dirty || opts.allow_dirty {
637642
let rev_obj = repo.revparse_single("HEAD")?;
638643
Ok(GitVcsInfo {
639644
sha1: rev_obj.id().to_string(),
645+
dirty,
640646
})
641647
} else {
642648
anyhow::bail!(

src/doc/man/cargo-package.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,8 @@ steps:
3131
executable binary or example target. {{man "cargo-install" 1}} will use the
3232
packaged lock file if the `--locked` flag is used.
3333
- A `.cargo_vcs_info.json` file is included that contains information
34-
about the current VCS checkout hash if available (not included with
35-
`--allow-dirty`).
34+
about the current VCS checkout hash if available, as well as a flag if the
35+
worktree is dirty.
3636
3. Extract the `.crate` file and build it to verify it can build.
3737
- This will rebuild your package from scratch to ensure that it can be
3838
built from a pristine state. The `--no-verify` flag can be used to skip
@@ -52,12 +52,16 @@ Will generate a `.cargo_vcs_info.json` in the following format
5252
```javascript
5353
{
5454
"git": {
55-
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302"
55+
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302",
56+
"dirty": true
5657
},
5758
"path_in_vcs": ""
5859
}
5960
```
6061

62+
`dirty` indicates that the Git worktree was dirty when the package
63+
was built.
64+
6165
`path_in_vcs` will be set to a repo-relative path for packages
6266
in subdirectories of the version control repository.
6367

src/doc/man/generated_txt/cargo-package.txt

+7-3
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,8 @@ DESCRIPTION
2929
packaged lock file if the --locked flag is used.
3030

3131
o A .cargo_vcs_info.json file is included that contains information
32-
about the current VCS checkout hash if available (not included
33-
with --allow-dirty).
32+
about the current VCS checkout hash if available, as well as a
33+
flag if the worktree is dirty.
3434

3535
3. Extract the .crate file and build it to verify it can build.
3636
o This will rebuild your package from scratch to ensure that it can
@@ -51,11 +51,15 @@ DESCRIPTION
5151

5252
{
5353
"git": {
54-
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302"
54+
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302",
55+
"dirty": true
5556
},
5657
"path_in_vcs": ""
5758
}
5859

60+
dirty indicates that the Git worktree was dirty when the package was
61+
built.
62+
5963
path_in_vcs will be set to a repo-relative path for packages in
6064
subdirectories of the version control repository.
6165

src/doc/src/commands/cargo-package.md

+7-3
Original file line numberDiff line numberDiff line change
@@ -26,8 +26,8 @@ steps:
2626
executable binary or example target. [cargo-install(1)](cargo-install.html) will use the
2727
packaged lock file if the `--locked` flag is used.
2828
- A `.cargo_vcs_info.json` file is included that contains information
29-
about the current VCS checkout hash if available (not included with
30-
`--allow-dirty`).
29+
about the current VCS checkout hash if available, as well as a flag if the
30+
worktree is dirty.
3131
3. Extract the `.crate` file and build it to verify it can build.
3232
- This will rebuild your package from scratch to ensure that it can be
3333
built from a pristine state. The `--no-verify` flag can be used to skip
@@ -47,12 +47,16 @@ Will generate a `.cargo_vcs_info.json` in the following format
4747
```javascript
4848
{
4949
"git": {
50-
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302"
50+
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302",
51+
"dirty": true
5152
},
5253
"path_in_vcs": ""
5354
}
5455
```
5556

57+
`dirty` indicates that the Git worktree was dirty when the package
58+
was built.
59+
5660
`path_in_vcs` will be set to a repo-relative path for packages
5761
in subdirectories of the version control repository.
5862

src/etc/man/cargo-package.1

+7-3
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,8 @@ packaged lock file if the \fB\-\-locked\fR flag is used.
4343
.sp
4444
.RS 4
4545
\h'-04'\(bu\h'+02'A \fB\&.cargo_vcs_info.json\fR file is included that contains information
46-
about the current VCS checkout hash if available (not included with
47-
\fB\-\-allow\-dirty\fR).
46+
about the current VCS checkout hash if available, as well as a flag if the
47+
worktree is dirty.
4848
.RE
4949
.RE
5050
.sp
@@ -74,13 +74,17 @@ Will generate a \fB\&.cargo_vcs_info.json\fR in the following format
7474
.nf
7575
{
7676
"git": {
77-
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302"
77+
"sha1": "aac20b6e7e543e6dd4118b246c77225e3a3a1302",
78+
"dirty": true
7879
},
7980
"path_in_vcs": ""
8081
}
8182
.fi
8283
.RE
8384
.sp
85+
\fBdirty\fR indicates that the Git worktree was dirty when the package
86+
was built.
87+
.sp
8488
\fBpath_in_vcs\fR will be set to a repo\-relative path for packages
8589
in subdirectories of the version control repository.
8690
.sp

tests/testsuite/git.rs

+1
Original file line numberDiff line numberDiff line change
@@ -2610,6 +2610,7 @@ fn include_overrides_gitignore() {
26102610
p.cargo("package --list --allow-dirty")
26112611
.with_stdout(
26122612
"\
2613+
.cargo_vcs_info.json
26132614
Cargo.toml
26142615
Cargo.toml.orig
26152616
ignored.txt

tests/testsuite/package.rs

+112
Original file line numberDiff line numberDiff line change
@@ -703,6 +703,7 @@ fn no_duplicates_from_modified_tracked_files() {
703703
p.cargo("package --list --allow-dirty")
704704
.with_stdout(
705705
"\
706+
.cargo_vcs_info.json
706707
Cargo.lock
707708
Cargo.toml
708709
Cargo.toml.orig
@@ -1011,6 +1012,7 @@ src/main.rs
10111012
.with_stderr("")
10121013
.with_stdout(
10131014
"\
1015+
.cargo_vcs_info.json
10141016
.gitignore
10151017
Cargo.lock
10161018
Cargo.toml
@@ -1171,6 +1173,111 @@ src/lib.rs
11711173
.run();
11721174
}
11731175

1176+
#[cargo_test]
1177+
fn issue_13695_allow_dirty_vcs_info() {
1178+
let p = project()
1179+
.file(
1180+
"Cargo.toml",
1181+
r#"
1182+
[package]
1183+
name = "foo"
1184+
version = "0.1.0"
1185+
edition = "2015"
1186+
description = "foo"
1187+
license = "foo"
1188+
documentation = "foo"
1189+
"#,
1190+
)
1191+
.file("src/lib.rs", "")
1192+
.build();
1193+
1194+
let repo = git::init(&p.root());
1195+
// Initial commit, with no files added.
1196+
git::commit(&repo);
1197+
1198+
// Allowing a dirty worktree results in the vcs file still being included.
1199+
p.cargo("package --allow-dirty").run();
1200+
1201+
let f = File::open(&p.root().join("target/package/foo-0.1.0.crate")).unwrap();
1202+
validate_crate_contents(
1203+
f,
1204+
"foo-0.1.0.crate",
1205+
&[
1206+
".cargo_vcs_info.json",
1207+
"Cargo.toml",
1208+
"Cargo.toml.orig",
1209+
"src/lib.rs",
1210+
],
1211+
&[(
1212+
".cargo_vcs_info.json",
1213+
r#"{
1214+
"git": {
1215+
"sha1": "[..]",
1216+
"dirty": true
1217+
},
1218+
"path_in_vcs": ""
1219+
}"#,
1220+
)],
1221+
);
1222+
1223+
// Listing provides a consistent result.
1224+
p.cargo("package --list --allow-dirty")
1225+
.with_stderr("")
1226+
.with_stdout(
1227+
"\
1228+
.cargo_vcs_info.json
1229+
Cargo.toml
1230+
Cargo.toml.orig
1231+
src/lib.rs
1232+
",
1233+
)
1234+
.run();
1235+
}
1236+
1237+
#[cargo_test]
1238+
fn issue_13695_allowing_dirty_vcs_info_but_clean() {
1239+
let p = project().build();
1240+
let _ = git::repo(&paths::root().join("foo"))
1241+
.file(
1242+
"Cargo.toml",
1243+
r#"
1244+
[package]
1245+
name = "foo"
1246+
version = "0.1.0"
1247+
edition = "2015"
1248+
description = "foo"
1249+
license = "foo"
1250+
documentation = "foo"
1251+
"#,
1252+
)
1253+
.file("src/lib.rs", "")
1254+
.build();
1255+
1256+
// Allowing a dirty worktree despite it being clean.
1257+
p.cargo("package --allow-dirty").run();
1258+
1259+
let f = File::open(&p.root().join("target/package/foo-0.1.0.crate")).unwrap();
1260+
validate_crate_contents(
1261+
f,
1262+
"foo-0.1.0.crate",
1263+
&[
1264+
".cargo_vcs_info.json",
1265+
"Cargo.toml",
1266+
"Cargo.toml.orig",
1267+
"src/lib.rs",
1268+
],
1269+
&[(
1270+
".cargo_vcs_info.json",
1271+
r#"{
1272+
"git": {
1273+
"sha1": "[..]"
1274+
},
1275+
"path_in_vcs": ""
1276+
}"#,
1277+
)],
1278+
);
1279+
}
1280+
11741281
#[cargo_test]
11751282
fn generated_manifest() {
11761283
let registry = registry::alt_init();
@@ -2333,6 +2440,7 @@ fn finds_git_in_parent() {
23332440
p.cargo("package --list --allow-dirty")
23342441
.with_stdout(
23352442
"\
2443+
.cargo_vcs_info.json
23362444
Cargo.toml
23372445
Cargo.toml.orig
23382446
ignoreme
@@ -2346,6 +2454,7 @@ src/lib.rs
23462454
p.cargo("package --list --allow-dirty")
23472455
.with_stdout(
23482456
"\
2457+
.cargo_vcs_info.json
23492458
.gitignore
23502459
Cargo.toml
23512460
Cargo.toml.orig
@@ -2359,6 +2468,7 @@ src/lib.rs
23592468
p.cargo("package --list --allow-dirty")
23602469
.with_stdout(
23612470
"\
2471+
.cargo_vcs_info.json
23622472
.gitignore
23632473
Cargo.toml
23642474
Cargo.toml.orig
@@ -2621,6 +2731,7 @@ fn deleted_git_working_tree() {
26212731
p.cargo("package --allow-dirty --list")
26222732
.with_stdout(
26232733
"\
2734+
.cargo_vcs_info.json
26242735
Cargo.lock
26252736
Cargo.toml
26262737
Cargo.toml.orig
@@ -2635,6 +2746,7 @@ src/main.rs
26352746
p.cargo("package --allow-dirty --list")
26362747
.with_stdout(
26372748
"\
2749+
.cargo_vcs_info.json
26382750
Cargo.lock
26392751
Cargo.toml
26402752
Cargo.toml.orig

tests/testsuite/publish_lockfile.rs

+1
Original file line numberDiff line numberDiff line change
@@ -249,6 +249,7 @@ fn note_resolve_changes() {
249249
[NOTE] package `multi v0.1.0` added to the packaged Cargo.lock file, was originally sourced from `[..]/foo/multi`
250250
[NOTE] package `patched v1.0.0` added to the packaged Cargo.lock file, was originally sourced from `[..]/foo/patched`
251251
[PACKAGED] [..] files, [..] ([..] compressed)
252+
[WARNING] no (git) Cargo.toml found at `target/tmp/[..]/foo/Cargo.toml` in workdir `[..]`
252253
",
253254
)
254255
.run();

0 commit comments

Comments
 (0)