Skip to content

Commit b396f2c

Browse files
committed
Auto merge of #14591 - epage:autolib, r=weihanglo
feat(toml): Add `autolib` ### What does this PR try to resolve? PR #5335 added `autobins`, etc for #5330. Nowhere in there is discussion of `autolib`. Cargo script disables support for additional build-targets by disabling discovery. Except we don't have a way to disable discovery of `autolib`, leading to #14476. By adding `autolib`, we can continue in that direction. This also allows us to bypass inferring of libs on published packages, like all other build-targets which were handled in #13849. Fixes #14476 ### How should we test and review this PR? ### Additional information As this seems fairly low controversy, this insta-stabilizes the field. In prior versions of Cargo, users will get an "unused manifest key" warning. For packages where this is set by `cargo publish`, the warning will be suppressed and things will work as normal. For `cargo vendor`, the same except there will be some churn in the vendored source as this field will now be set. For local development, it should be rare to set `autolib` so the lack of error by discovering a file when this is set shouldn't be a problem.
2 parents a3b35a0 + 5e35e27 commit b396f2c

19 files changed

+160
-47
lines changed

Cargo.lock

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ cargo-platform = { path = "crates/cargo-platform", version = "0.1.5" }
3333
cargo-test-macro = { version = "0.3.0", path = "crates/cargo-test-macro" }
3434
cargo-test-support = { version = "0.5.0", path = "crates/cargo-test-support" }
3535
cargo-util = { version = "0.2.14", path = "crates/cargo-util" }
36-
cargo-util-schemas = { version = "0.6.0", path = "crates/cargo-util-schemas" }
36+
cargo-util-schemas = { version = "0.7.0", path = "crates/cargo-util-schemas" }
3737
cargo_metadata = "0.18.1"
3838
clap = "4.5.18"
3939
clap_complete = { version = "4.5.29", features = ["unstable-dynamic"] }

crates/cargo-util-schemas/Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[package]
22
name = "cargo-util-schemas"
3-
version = "0.6.1"
3+
version = "0.7.0"
44
rust-version = "1.81" # MSRV:1
55
edition.workspace = true
66
license.workspace = true

crates/cargo-util-schemas/src/manifest/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,7 @@ pub struct TomlPackage {
173173
pub publish: Option<InheritableVecStringOrBool>,
174174
pub workspace: Option<String>,
175175
pub im_a_teapot: Option<bool>,
176+
pub autolib: Option<bool>,
176177
pub autobins: Option<bool>,
177178
pub autoexamples: Option<bool>,
178179
pub autotests: Option<bool>,
@@ -217,6 +218,7 @@ impl TomlPackage {
217218
publish: None,
218219
workspace: None,
219220
im_a_teapot: None,
221+
autolib: None,
220222
autobins: None,
221223
autoexamples: None,
222224
autotests: None,

src/cargo/util/toml/embedded.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,13 @@ use crate::GlobalContext;
88

99
const DEFAULT_EDITION: crate::core::features::Edition =
1010
crate::core::features::Edition::LATEST_STABLE;
11-
const AUTO_FIELDS: &[&str] = &["autobins", "autoexamples", "autotests", "autobenches"];
11+
const AUTO_FIELDS: &[&str] = &[
12+
"autolib",
13+
"autobins",
14+
"autoexamples",
15+
"autotests",
16+
"autobenches",
17+
];
1218

1319
pub(super) fn expand_manifest(
1420
content: &str,
@@ -289,6 +295,7 @@ path = "/home/me/test.rs"
289295
autobenches = false
290296
autobins = false
291297
autoexamples = false
298+
autolib = false
292299
autotests = false
293300
build = false
294301
edition = "2021"
@@ -324,6 +331,7 @@ time = "0.1.25"
324331
autobenches = false
325332
autobins = false
326333
autoexamples = false
334+
autolib = false
327335
autotests = false
328336
build = false
329337
edition = "2021"
@@ -359,6 +367,7 @@ time = "0.1.25"
359367
autobenches = false
360368
autobins = false
361369
autoexamples = false
370+
autolib = false
362371
autotests = false
363372
build = false
364373
edition = "2021"

src/cargo/util/toml/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -334,6 +334,7 @@ fn normalize_toml(
334334
package_root,
335335
&original_package.name,
336336
edition,
337+
original_package.autolib,
337338
warnings,
338339
)?;
339340
normalized_toml.bin = Some(targets::normalize_bins(
@@ -624,6 +625,7 @@ fn normalize_package_toml<'a>(
624625
.map(manifest::InheritableField::Value),
625626
workspace: original_package.workspace.clone(),
626627
im_a_teapot: original_package.im_a_teapot.clone(),
628+
autolib: Some(false),
627629
autobins: Some(false),
628630
autoexamples: Some(false),
629631
autotests: Some(false),

src/cargo/util/toml/targets.rs

+58-35
Original file line numberDiff line numberDiff line change
@@ -129,48 +129,63 @@ pub fn normalize_lib(
129129
package_root: &Path,
130130
package_name: &str,
131131
edition: Edition,
132+
autodiscover: Option<bool>,
132133
warnings: &mut Vec<String>,
133134
) -> CargoResult<Option<TomlLibTarget>> {
134-
let inferred = inferred_lib(package_root);
135-
let lib = original_lib.cloned().or_else(|| {
136-
inferred.as_ref().map(|lib| TomlTarget {
137-
path: Some(PathValue(lib.clone())),
138-
..TomlTarget::new()
139-
})
140-
});
141-
let Some(mut lib) = lib else { return Ok(None) };
142-
lib.name
143-
.get_or_insert_with(|| package_name.replace("-", "_"));
135+
if is_normalized(original_lib, autodiscover) {
136+
let Some(lib) = original_lib.cloned() else {
137+
return Ok(None);
138+
};
144139

145-
// Check early to improve error messages
146-
validate_lib_name(&lib, warnings)?;
140+
// Check early to improve error messages
141+
validate_lib_name(&lib, warnings)?;
147142

148-
validate_proc_macro(&lib, "library", edition, warnings)?;
149-
validate_crate_types(&lib, "library", edition, warnings)?;
143+
validate_proc_macro(&lib, "library", edition, warnings)?;
144+
validate_crate_types(&lib, "library", edition, warnings)?;
150145

151-
if lib.path.is_none() {
152-
if let Some(inferred) = inferred {
153-
lib.path = Some(PathValue(inferred));
154-
} else {
155-
let name = name_or_panic(&lib);
156-
let legacy_path = Path::new("src").join(format!("{name}.rs"));
157-
if edition == Edition::Edition2015 && package_root.join(&legacy_path).exists() {
158-
warnings.push(format!(
159-
"path `{}` was erroneously implicitly accepted for library `{name}`,\n\
160-
please rename the file to `src/lib.rs` or set lib.path in Cargo.toml",
161-
legacy_path.display(),
162-
));
163-
lib.path = Some(PathValue(legacy_path));
146+
Ok(Some(lib))
147+
} else {
148+
let inferred = inferred_lib(package_root);
149+
let lib = original_lib.cloned().or_else(|| {
150+
inferred.as_ref().map(|lib| TomlTarget {
151+
path: Some(PathValue(lib.clone())),
152+
..TomlTarget::new()
153+
})
154+
});
155+
let Some(mut lib) = lib else { return Ok(None) };
156+
lib.name
157+
.get_or_insert_with(|| package_name.replace("-", "_"));
158+
159+
// Check early to improve error messages
160+
validate_lib_name(&lib, warnings)?;
161+
162+
validate_proc_macro(&lib, "library", edition, warnings)?;
163+
validate_crate_types(&lib, "library", edition, warnings)?;
164+
165+
if lib.path.is_none() {
166+
if let Some(inferred) = inferred {
167+
lib.path = Some(PathValue(inferred));
164168
} else {
165-
anyhow::bail!(
166-
"can't find library `{name}`, \
169+
let name = name_or_panic(&lib);
170+
let legacy_path = Path::new("src").join(format!("{name}.rs"));
171+
if edition == Edition::Edition2015 && package_root.join(&legacy_path).exists() {
172+
warnings.push(format!(
173+
"path `{}` was erroneously implicitly accepted for library `{name}`,\n\
174+
please rename the file to `src/lib.rs` or set lib.path in Cargo.toml",
175+
legacy_path.display(),
176+
));
177+
lib.path = Some(PathValue(legacy_path));
178+
} else {
179+
anyhow::bail!(
180+
"can't find library `{name}`, \
167181
rename file to `src/lib.rs` or specify lib.path",
168-
)
182+
)
183+
}
169184
}
170185
}
171-
}
172186

173-
Ok(Some(lib))
187+
Ok(Some(lib))
188+
}
174189
}
175190

176191
#[tracing::instrument(skip_all)]
@@ -239,7 +254,7 @@ pub fn normalize_bins(
239254
errors: &mut Vec<String>,
240255
has_lib: bool,
241256
) -> CargoResult<Vec<TomlBinTarget>> {
242-
if is_normalized(toml_bins, autodiscover) {
257+
if are_normalized(toml_bins, autodiscover) {
243258
let toml_bins = toml_bins.cloned().unwrap_or_default();
244259
for bin in &toml_bins {
245260
validate_bin_name(bin, warnings)?;
@@ -526,7 +541,15 @@ fn to_bench_targets(
526541
Ok(result)
527542
}
528543

529-
fn is_normalized(toml_targets: Option<&Vec<TomlTarget>>, autodiscover: Option<bool>) -> bool {
544+
fn is_normalized(toml_target: Option<&TomlTarget>, autodiscover: Option<bool>) -> bool {
545+
are_normalized_(toml_target.map(std::slice::from_ref), autodiscover)
546+
}
547+
548+
fn are_normalized(toml_targets: Option<&Vec<TomlTarget>>, autodiscover: Option<bool>) -> bool {
549+
are_normalized_(toml_targets.map(|v| v.as_slice()), autodiscover)
550+
}
551+
552+
fn are_normalized_(toml_targets: Option<&[TomlTarget]>, autodiscover: Option<bool>) -> bool {
530553
if autodiscover != Some(false) {
531554
return false;
532555
}
@@ -579,7 +602,7 @@ fn normalize_targets_with_legacy_path(
579602
legacy_path: &mut dyn FnMut(&TomlTarget) -> Option<PathBuf>,
580603
autodiscover_flag_name: &str,
581604
) -> CargoResult<Vec<TomlTarget>> {
582-
if is_normalized(toml_targets, autodiscover) {
605+
if are_normalized(toml_targets, autodiscover) {
583606
let toml_targets = toml_targets.cloned().unwrap_or_default();
584607
for target in &toml_targets {
585608
// Check early to improve error messages

src/doc/src/reference/cargo-targets.md

+5-1
Original file line numberDiff line numberDiff line change
@@ -323,13 +323,14 @@ configuration tables, such as `[lib]`, `[[bin]]`, `[[test]]`, `[[bench]]`, or
323323
standard directory layout.
324324

325325
The automatic target discovery can be disabled so that only manually
326-
configured targets will be built. Setting the keys `autobins`, `autoexamples`,
326+
configured targets will be built. Setting the keys `autolib`, `autobins`, `autoexamples`,
327327
`autotests`, or `autobenches` to `false` in the `[package]` section will
328328
disable auto-discovery of the corresponding target type.
329329

330330
```toml
331331
[package]
332332
# ...
333+
autolib = false
333334
autobins = false
334335
autoexamples = false
335336
autotests = false
@@ -363,6 +364,9 @@ autobins = false
363364
> is `false` if at least one target is manually defined in `Cargo.toml`.
364365
> Beginning with the 2018 edition, the default is always `true`.
365366
367+
> **MSRV:** Respected as of 1.27 for `autobins`, `autoexamples`, `autotests`, and `autobenches`
368+
369+
> **MSRV:** Respected as of 1.83 for `autolib`
366370
367371
[Build cache]: ../guide/build-cache.md
368372
[Rust Edition]: ../../edition-guide/index.html

src/doc/src/reference/manifest.md

+1
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ Every manifest file consists of the following sections:
3030
* [`publish`](#the-publish-field) --- Can be used to prevent publishing the package.
3131
* [`metadata`](#the-metadata-table) --- Extra settings for external tools.
3232
* [`default-run`](#the-default-run-field) --- The default binary to run by [`cargo run`].
33+
* [`autolib`](cargo-targets.md#target-auto-discovery) --- Disables library auto discovery.
3334
* [`autobins`](cargo-targets.md#target-auto-discovery) --- Disables binary auto discovery.
3435
* [`autoexamples`](cargo-targets.md#target-auto-discovery) --- Disables example auto discovery.
3536
* [`autotests`](cargo-targets.md#target-auto-discovery) --- Disables test auto discovery.

src/doc/src/reference/unstable.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -1314,7 +1314,7 @@ Inferred / defaulted manifest fields:
13141314

13151315
Disallowed manifest fields:
13161316
- `[workspace]`, `[lib]`, `[[bin]]`, `[[example]]`, `[[test]]`, `[[bench]]`
1317-
- `package.workspace`, `package.build`, `package.links`, `package.autobins`, `package.autoexamples`, `package.autotests`, `package.autobenches`
1317+
- `package.workspace`, `package.build`, `package.links`, `package.autolib`, `package.autobins`, `package.autoexamples`, `package.autotests`, `package.autobenches`
13181318

13191319
The default `CARGO_TARGET_DIR` for single-file packages is at `$CARGO_HOME/target/<hash>`:
13201320
- Avoid conflicts from multiple single-file packages being in the same directory

tests/testsuite/artifact_dep.rs

+1
Original file line numberDiff line numberDiff line change
@@ -2272,6 +2272,7 @@ name = "foo"
22722272
version = "0.1.0"
22732273
authors = []
22742274
build = false
2275+
autolib = false
22752276
autobins = false
22762277
autoexamples = false
22772278
autotests = false

tests/testsuite/features2.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1800,6 +1800,7 @@ name = "a"
18001800
version = "0.1.0"
18011801
authors = ["Zzz"]
18021802
build = false
1803+
autolib = false
18031804
autobins = false
18041805
autoexamples = false
18051806
autotests = false

tests/testsuite/features_namespaced.rs

+2
Original file line numberDiff line numberDiff line change
@@ -985,6 +985,7 @@ edition = "2015"
985985
name = "foo"
986986
version = "0.1.0"
987987
build = false
988+
autolib = false
988989
autobins = false
989990
autoexamples = false
990991
autotests = false
@@ -1112,6 +1113,7 @@ edition = "2015"
11121113
name = "foo"
11131114
version = "0.1.0"
11141115
build = false
1116+
autolib = false
11151117
autobins = false
11161118
autoexamples = false
11171119
autotests = false

tests/testsuite/inheritable_workspace_fields.rs

+5
Original file line numberDiff line numberDiff line change
@@ -224,6 +224,7 @@ include = [
224224
"Cargo.toml",
225225
]
226226
publish = true
227+
autolib = false
227228
autobins = false
228229
autoexamples = false
229230
autotests = false
@@ -394,6 +395,7 @@ name = "bar"
394395
version = "0.2.0"
395396
authors = []
396397
build = false
398+
autolib = false
397399
autobins = false
398400
autoexamples = false
399401
autotests = false
@@ -533,6 +535,7 @@ name = "bar"
533535
version = "0.2.0"
534536
authors = []
535537
build = false
538+
autolib = false
536539
autobins = false
537540
autoexamples = false
538541
autotests = false
@@ -793,6 +796,7 @@ include = [
793796
"README.md",
794797
]
795798
publish = true
799+
autolib = false
796800
autobins = false
797801
autoexamples = false
798802
autotests = false
@@ -966,6 +970,7 @@ name = "bar"
966970
version = "0.2.0"
967971
authors = []
968972
build = false
973+
autolib = false
969974
autobins = false
970975
autoexamples = false
971976
autotests = false

0 commit comments

Comments
 (0)