Skip to content

Commit e539380

Browse files
committed
feat: Stabilize lints
Fixes #12115
1 parent 4518131 commit e539380

File tree

7 files changed

+118
-294
lines changed

7 files changed

+118
-294
lines changed

src/cargo/ops/cargo_new.rs

+4-5
Original file line numberDiff line numberDiff line change
@@ -817,11 +817,10 @@ fn mk(config: &Config, opts: &MkOptions<'_>) -> CargoResult<()> {
817817
}
818818

819819
// Try to inherit the workspace lints key if it exists.
820-
if config.cli_unstable().lints
821-
&& workspace_document
822-
.get("workspace")
823-
.and_then(|workspace| workspace.get("lints"))
824-
.is_some()
820+
if workspace_document
821+
.get("workspace")
822+
.and_then(|workspace| workspace.get("lints"))
823+
.is_some()
825824
{
826825
let mut table = toml_edit::Table::new();
827826
table["workspace"] = toml_edit::value(true);

src/cargo/util/toml/mod.rs

+13-58
Original file line numberDiff line numberDiff line change
@@ -339,7 +339,7 @@ pub struct TomlManifest {
339339
patch: Option<BTreeMap<String, BTreeMap<String, TomlDependency>>>,
340340
workspace: Option<TomlWorkspace>,
341341
badges: Option<MaybeWorkspaceBtreeMap>,
342-
lints: Option<toml::Value>,
342+
lints: Option<MaybeWorkspaceLints>,
343343
}
344344

345345
#[derive(Deserialize, Serialize, Clone, Debug, Default)]
@@ -1456,7 +1456,7 @@ pub struct TomlWorkspace {
14561456
// Properties that can be inherited by members.
14571457
package: Option<InheritableFields>,
14581458
dependencies: Option<BTreeMap<String, TomlDependency>>,
1459-
lints: Option<toml::Value>,
1459+
lints: Option<TomlLints>,
14601460

14611461
// Note that this field must come last due to the way toml serialization
14621462
// works which requires tables to be emitted after all values.
@@ -1882,7 +1882,7 @@ impl TomlManifest {
18821882
let mut inheritable = toml_config.package.clone().unwrap_or_default();
18831883
inheritable.update_ws_path(package_root.to_path_buf());
18841884
inheritable.update_deps(toml_config.dependencies.clone());
1885-
let lints = parse_unstable_lints(toml_config.lints.clone(), config, &mut warnings)?;
1885+
let lints = toml_config.lints.clone();
18861886
let lints = verify_lints(lints)?;
18871887
inheritable.update_lints(lints);
18881888
if let Some(ws_deps) = &inheritable.dependencies {
@@ -2143,10 +2143,11 @@ impl TomlManifest {
21432143
&inherit_cell,
21442144
)?;
21452145

2146-
let lints =
2147-
parse_unstable_lints::<MaybeWorkspaceLints>(me.lints.clone(), config, cx.warnings)?
2148-
.map(|mw| mw.resolve(|| inherit()?.lints()))
2149-
.transpose()?;
2146+
let lints = me
2147+
.lints
2148+
.clone()
2149+
.map(|mw| mw.resolve(|| inherit()?.lints()))
2150+
.transpose()?;
21502151
let lints = verify_lints(lints)?;
21512152
let default = TomlLints::default();
21522153
let rustflags = lints_to_rustflags(lints.as_ref().unwrap_or(&default));
@@ -2447,7 +2448,10 @@ impl TomlManifest {
24472448
.badges
24482449
.as_ref()
24492450
.map(|_| MaybeWorkspace::Defined(metadata.badges.clone())),
2450-
lints: lints.map(|lints| toml::Value::try_from(lints).unwrap()),
2451+
lints: lints.map(|lints| MaybeWorkspaceLints {
2452+
workspace: false,
2453+
lints,
2454+
}),
24512455
};
24522456
let mut manifest = Manifest::new(
24532457
summary,
@@ -2579,7 +2583,7 @@ impl TomlManifest {
25792583
let mut inheritable = toml_config.package.clone().unwrap_or_default();
25802584
inheritable.update_ws_path(root.to_path_buf());
25812585
inheritable.update_deps(toml_config.dependencies.clone());
2582-
let lints = parse_unstable_lints(toml_config.lints.clone(), config, &mut warnings)?;
2586+
let lints = toml_config.lints.clone();
25832587
let lints = verify_lints(lints)?;
25842588
inheritable.update_lints(lints);
25852589
let ws_root_config = WorkspaceRootConfig::new(
@@ -2726,55 +2730,6 @@ impl TomlManifest {
27262730
}
27272731
}
27282732

2729-
fn parse_unstable_lints<T: Deserialize<'static>>(
2730-
lints: Option<toml::Value>,
2731-
config: &Config,
2732-
warnings: &mut Vec<String>,
2733-
) -> CargoResult<Option<T>> {
2734-
let Some(lints) = lints else {
2735-
return Ok(None);
2736-
};
2737-
2738-
if !config.cli_unstable().lints {
2739-
warn_for_lint_feature(config, warnings);
2740-
return Ok(None);
2741-
}
2742-
2743-
lints.try_into().map(Some).map_err(|err| err.into())
2744-
}
2745-
2746-
fn warn_for_lint_feature(config: &Config, warnings: &mut Vec<String>) {
2747-
use std::fmt::Write as _;
2748-
2749-
let key_name = "lints";
2750-
let feature_name = "lints";
2751-
2752-
let mut message = String::new();
2753-
2754-
let _ = write!(
2755-
message,
2756-
"unused manifest key `{key_name}` (may be supported in a future version)"
2757-
);
2758-
if config.nightly_features_allowed {
2759-
let _ = write!(
2760-
message,
2761-
"
2762-
2763-
consider passing `-Z{feature_name}` to enable this feature."
2764-
);
2765-
} else {
2766-
let _ = write!(
2767-
message,
2768-
"
2769-
2770-
this Cargo does not support nightly features, but if you
2771-
switch to nightly channel you can pass
2772-
`-Z{feature_name}` to enable this feature.",
2773-
);
2774-
}
2775-
warnings.push(message);
2776-
}
2777-
27782733
fn verify_lints(lints: Option<TomlLints>) -> CargoResult<Option<TomlLints>> {
27792734
let Some(lints) = lints else {
27802735
return Ok(None);

src/doc/src/reference/manifest.md

+41
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@ Every manifest file consists of the following sections:
4848
* [`[target]`](specifying-dependencies.md#platform-specific-dependencies) --- Platform-specific dependencies.
4949
* [`[badges]`](#the-badges-section) --- Badges to display on a registry.
5050
* [`[features]`](features.md) --- Conditional compilation features.
51+
* [`[lints]`](#the-lints-section) --- Configure linters for this package.
5152
* [`[patch]`](overriding-dependencies.md#the-patch-section) --- Override dependencies.
5253
* [`[replace]`](overriding-dependencies.md#the-replace-section) --- Override dependencies (deprecated).
5354
* [`[profile]`](profiles.md) --- Compiler settings and optimizations.
@@ -530,6 +531,46 @@ both `src/bin/a.rs` and `src/bin/b.rs`:
530531
default-run = "a"
531532
```
532533

534+
#### The `lints` section
535+
536+
Override the default level of lints from different tools by assigning them to a new level in a
537+
table, for example:
538+
```toml
539+
[lints.rust]
540+
unsafe_code = "forbid"
541+
```
542+
543+
This is short-hand for:
544+
```toml
545+
[lints.rust]
546+
unsafe_code = { level = "forbid", priority = 0 }
547+
```
548+
549+
`level` corresponds to the lint levels in `rustc`:
550+
- `forbid`
551+
- `deny`
552+
- `warn`
553+
- `allow`
554+
555+
`priority` is a signed integer that controls which lints or lint groups override other lint groups:
556+
- lower (particularly negative) numbers have lower priority, being overridden
557+
by higher numbers, and show up first on the command-line to tools like
558+
`rustc`
559+
560+
To know which table under `[lints]` a particular lint belongs under, it is the part before `::` in the lint
561+
name. If there isn't a `::`, then the tool is `rust`. For example a warning
562+
about `unsafe_code` would be `lints.rust.unsafe_code` but a lint about
563+
`clippy::enum_glob_use` would be `lints.clippy.enum_glob_use`.
564+
565+
For example:
566+
```toml
567+
[lints.rust]
568+
unsafe_code = "forbid"
569+
570+
[lints.clippy]
571+
enum_glob_use = "deny"
572+
```
573+
533574
## The `[badges]` section
534575

535576
The `[badges]` section is for specifying status badges that can be displayed

src/doc/src/reference/unstable.md

+5-93
Original file line numberDiff line numberDiff line change
@@ -93,7 +93,6 @@ For the latest nightly, see the [nightly version] of this page.
9393
* [codegen-backend](#codegen-backend) --- Select the codegen backend used by rustc.
9494
* [per-package-target](#per-package-target) --- Sets the `--target` to use for each individual package.
9595
* [artifact dependencies](#artifact-dependencies) --- Allow build artifacts to be included into other build artifacts and build them for different targets.
96-
* [`[lints]`](#lints) --- Configure lint levels for various linter tools.
9796
* Information and metadata
9897
* [Build-plan](#build-plan) --- Emits JSON information on which commands will be run.
9998
* [unit-graph](#unit-graph) --- Emits JSON for Cargo's internal graph structure.
@@ -1571,100 +1570,8 @@ Differences between `cargo run --manifest-path <path>` and `cargo <path>`
15711570
- `cargo <path>` runs with the config for `<path>` and not the current dir, more like `cargo install --path <path>`
15721571
- `cargo <path>` is at a verbosity level below the normal default. Pass `-v` to get normal output.
15731572

1574-
## `[lints]`
1575-
1576-
* Tracking Issue: [#12115](https://github.com/rust-lang/cargo/issues/12115)
1577-
1578-
A new `lints` table would be added to configure lints:
1579-
```toml
1580-
[lints.rust]
1581-
unsafe = "forbid"
1582-
```
1583-
and `cargo` would pass these along as flags to `rustc`, `clippy`, or other lint tools when `-Zlints` is used.
1584-
1585-
This would work with
1586-
[RFC 2906 `workspace-deduplicate`](https://rust-lang.github.io/rfcs/2906-cargo-workspace-deduplicate.html):
1587-
```toml
1588-
[lints]
1589-
workspace = true
1590-
1591-
[workspace.lints.rust]
1592-
unsafe = "forbid"
1593-
```
1594-
15951573
### Documentation Updates
15961574

1597-
#### The `lints` section
1598-
1599-
*as a new ["Manifest Format" entry](./manifest.html#the-manifest-format)*
1600-
1601-
Override the default level of lints from different tools by assigning them to a new level in a
1602-
table, for example:
1603-
```toml
1604-
[lints.rust]
1605-
unsafe = "forbid"
1606-
```
1607-
1608-
This is short-hand for:
1609-
```toml
1610-
[lints.rust]
1611-
unsafe = { level = "forbid", priority = 0 }
1612-
```
1613-
1614-
`level` corresponds to the lint levels in `rustc`:
1615-
- `forbid`
1616-
- `deny`
1617-
- `warn`
1618-
- `allow`
1619-
1620-
`priority` is a signed integer that controls which lints or lint groups override other lint groups:
1621-
- lower (particularly negative) numbers have lower priority, being overridden
1622-
by higher numbers, and show up first on the command-line to tools like
1623-
`rustc`
1624-
1625-
To know which table under `[lints]` a particular lint belongs under, it is the part before `::` in the lint
1626-
name. If there isn't a `::`, then the tool is `rust`. For example a warning
1627-
about `unsafe` would be `lints.rust.unsafe` but a lint about
1628-
`clippy::enum_glob_use` would be `lints.clippy.enum_glob_use`.
1629-
1630-
For example:
1631-
```toml
1632-
[lints.rust]
1633-
unsafe = "forbid"
1634-
1635-
[lints.clippy]
1636-
enum_glob_use = "deny"
1637-
```
1638-
1639-
#### The `lints` table
1640-
1641-
*as a new [`[workspace]` entry](./workspaces.html#the-workspace-section)*
1642-
1643-
The `workspace.lints` table is where you define lint configuration to be inherited by members of a workspace.
1644-
1645-
Specifying a workspace lint configuration is similar to package lints.
1646-
1647-
Example:
1648-
1649-
```toml
1650-
# [PROJECT_DIR]/Cargo.toml
1651-
[workspace]
1652-
members = ["crates/*"]
1653-
1654-
[workspace.lints.rust]
1655-
unsafe = "forbid"
1656-
```
1657-
1658-
```toml
1659-
# [PROJECT_DIR]/crates/bar/Cargo.toml
1660-
[package]
1661-
name = "bar"
1662-
version = "0.1.0"
1663-
1664-
[lints]
1665-
workspace = true
1666-
```
1667-
16681575
# Stabilized and removed features
16691576

16701577
## Compile progress
@@ -1886,3 +1793,8 @@ for more information about the working directory for compiling and running tests
18861793
The `--keep-going` option has been stabilized in the 1.74 release. See the
18871794
[`--keep-going` flag](../commands/cargo-build.html#option-cargo-build---keep-going)
18881795
in `cargo build` as an example for more details.
1796+
1797+
## `[lints]`
1798+
1799+
[`[lints]`](manifest.html#the-lints-section) (enabled via `-Zlints`) has been stabilized in the 1.74 release.
1800+

src/doc/src/reference/workspaces.md

+28
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@ In the `Cargo.toml`, the `[workspace]` table supports the following sections:
2424
* [`default-members`](#the-default-members-field) --- Packages to operate on when a specific package wasn't selected.
2525
* [`package`](#the-package-table) --- Keys for inheriting in packages.
2626
* [`dependencies`](#the-dependencies-table) --- Keys for inheriting in package dependencies.
27+
* [`lints`](#the-lints-table) --- Keys for inheriting in package lints.
2728
* [`metadata`](#the-metadata-table) --- Extra settings for external tools.
2829
* [`[patch]`](overriding-dependencies.md#the-patch-section) --- Override dependencies.
2930
* [`[replace]`](overriding-dependencies.md#the-replace-section) --- Override dependencies (deprecated).
@@ -222,6 +223,33 @@ cc.workspace = true
222223
rand.workspace = true
223224
```
224225

226+
## The `lints` table
227+
228+
The `workspace.lints` table is where you define lint configuration to be inherited by members of a workspace.
229+
230+
Specifying a workspace lint configuration is similar to package lints.
231+
232+
Example:
233+
234+
```toml
235+
# [PROJECT_DIR]/Cargo.toml
236+
[workspace]
237+
members = ["crates/*"]
238+
239+
[workspace.lints.rust]
240+
unsafe_code = "forbid"
241+
```
242+
243+
```toml
244+
# [PROJECT_DIR]/crates/bar/Cargo.toml
245+
[package]
246+
name = "bar"
247+
version = "0.1.0"
248+
249+
[lints]
250+
workspace = true
251+
```
252+
225253
## The `metadata` table
226254

227255
The `workspace.metadata` table is ignored by Cargo and will not be warned

tests/testsuite/cargo_new/inherit_workspace_lints/mod.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,6 @@
11
use cargo_test_support::compare::assert_ui;
22
use cargo_test_support::curr_dir;
33
use cargo_test_support::CargoCommand;
4-
use cargo_test_support::ChannelChanger;
54
use cargo_test_support::Project;
65

76
#[cargo_test]
@@ -12,9 +11,8 @@ fn case() {
1211

1312
snapbox::cmd::Command::cargo_ui()
1413
.arg("new")
15-
.args(["crates/foo", "-Zlints"])
14+
.args(["crates/foo"])
1615
.current_dir(cwd)
17-
.masquerade_as_nightly_cargo(&["lints"])
1816
.assert()
1917
.success()
2018
.stdout_matches_path(curr_dir!().join("stdout.log"))

0 commit comments

Comments
 (0)