Skip to content

Commit 064184a

Browse files
committed
feat(fix): Add features for unused deps on Edition 2024
1 parent 87915e7 commit 064184a

File tree

2 files changed

+177
-1
lines changed

2 files changed

+177
-1
lines changed

src/cargo/ops/fix.rs

+42-1
Original file line numberDiff line numberDiff line change
@@ -53,12 +53,13 @@ use tracing::{debug, trace, warn};
5353
use crate::core::compiler::RustcTargetData;
5454
use crate::core::resolver::features::{DiffMap, FeatureOpts, FeatureResolver, FeaturesFor};
5555
use crate::core::resolver::{HasDevUnits, Resolve, ResolveBehavior};
56-
use crate::core::PackageIdSpecQuery as _;
5756
use crate::core::{Edition, MaybePackage, Package, PackageId, Workspace};
57+
use crate::core::{FeatureValue, PackageIdSpecQuery as _};
5858
use crate::ops::resolve::WorkspaceResolve;
5959
use crate::ops::{self, CompileOptions};
6060
use crate::util::diagnostic_server::{Message, RustfixDiagnosticServer};
6161
use crate::util::errors::CargoResult;
62+
use crate::util::interning::InternedString;
6263
use crate::util::GlobalContext;
6364
use crate::util::{existing_vcs_repo, LockServer, LockServerClient};
6465
use crate::{drop_eprint, drop_eprintln};
@@ -253,6 +254,7 @@ fn migrate_manifests(ws: &Workspace<'_>, pkgs: &[&Package]) -> CargoResult<()> {
253254
let mut fixes = 0;
254255

255256
let root = document.as_table_mut();
257+
fixes += add_feature_for_unused_deps(pkg, root);
256258
if rename_table(root, "project", "package") {
257259
fixes += 1;
258260
}
@@ -287,6 +289,45 @@ fn rename_table(parent: &mut dyn toml_edit::TableLike, old: &str, new: &str) ->
287289
true
288290
}
289291

292+
fn add_feature_for_unused_deps(pkg: &Package, parent: &mut dyn toml_edit::TableLike) -> usize {
293+
let manifest = pkg.manifest();
294+
295+
let activated_opt_deps = manifest
296+
.resolved_toml()
297+
.features()
298+
.map(|map| {
299+
map.values()
300+
.flatten()
301+
.filter_map(|f| match FeatureValue::new(InternedString::new(f)) {
302+
FeatureValue::Dep { dep_name } => Some(dep_name.as_str()),
303+
_ => None,
304+
})
305+
.collect::<HashSet<_>>()
306+
})
307+
.unwrap_or_default();
308+
309+
let mut fixes = 0;
310+
for dep in manifest.dependencies() {
311+
let dep_name_in_toml = dep.name_in_toml();
312+
if dep.is_optional() && !activated_opt_deps.contains(dep_name_in_toml.as_str()) {
313+
fixes += 1;
314+
if let Some(features) = parent
315+
.entry("features")
316+
.or_insert(toml_edit::table())
317+
.as_table_like_mut()
318+
{
319+
features.insert(
320+
dep_name_in_toml.as_str(),
321+
toml_edit::Item::Value(toml_edit::Value::Array(toml_edit::Array::from_iter(
322+
&[format!("dep:{}", dep_name_in_toml)],
323+
))),
324+
);
325+
}
326+
}
327+
}
328+
fixes
329+
}
330+
290331
fn check_resolver_change(ws: &Workspace<'_>, opts: &FixOptions) -> CargoResult<()> {
291332
let root = ws.root_maybe();
292333
match root {

tests/testsuite/fix.rs

+135
Original file line numberDiff line numberDiff line change
@@ -2049,3 +2049,138 @@ edition = "2021"
20492049
"#
20502050
);
20512051
}
2052+
2053+
#[cargo_test]
2054+
fn add_feature_for_unused_dep() {
2055+
Package::new("bar", "0.1.0").publish();
2056+
Package::new("baz", "0.1.0").publish();
2057+
Package::new("target-dep", "0.1.0").publish();
2058+
let p = project()
2059+
.file(
2060+
"Cargo.toml",
2061+
r#"
2062+
[package]
2063+
name = "foo"
2064+
version = "0.1.0"
2065+
edition = "2021"
2066+
2067+
[dependencies]
2068+
bar = { version = "0.1.0", optional = true }
2069+
2070+
[build-dependencies]
2071+
baz = { version = "0.1.0", optional = true }
2072+
2073+
[target.'cfg(target_os = "linux")'.dependencies]
2074+
target-dep = { version = "0.1.0", optional = true }
2075+
"#,
2076+
)
2077+
.file("src/lib.rs", "")
2078+
.build();
2079+
2080+
p.cargo("fix --edition --allow-no-vcs")
2081+
.masquerade_as_nightly_cargo(&["edition2024"])
2082+
.with_stderr(
2083+
"\
2084+
[MIGRATING] Cargo.toml from 2021 edition to 2024
2085+
[FIXED] Cargo.toml (3 fixes)
2086+
[UPDATING] `dummy-registry` index
2087+
[LOCKING] 4 packages to latest compatible versions
2088+
[CHECKING] foo v0.1.0 ([CWD])
2089+
[MIGRATING] src/lib.rs from 2021 edition to 2024
2090+
[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [..]s
2091+
",
2092+
)
2093+
.run();
2094+
assert_eq!(
2095+
p.read_file("Cargo.toml"),
2096+
r#"
2097+
[package]
2098+
name = "foo"
2099+
version = "0.1.0"
2100+
edition = "2021"
2101+
2102+
[dependencies]
2103+
bar = { version = "0.1.0", optional = true }
2104+
2105+
[build-dependencies]
2106+
baz = { version = "0.1.0", optional = true }
2107+
2108+
[target.'cfg(target_os = "linux")'.dependencies]
2109+
target-dep = { version = "0.1.0", optional = true }
2110+
2111+
[features]
2112+
bar = ["dep:bar"]
2113+
baz = ["dep:baz"]
2114+
target-dep = ["dep:target-dep"]
2115+
"#
2116+
);
2117+
}
2118+
2119+
#[cargo_test]
2120+
fn add_feature_for_unused_dep_existing_table() {
2121+
Package::new("bar", "0.1.0").publish();
2122+
Package::new("baz", "0.1.0").publish();
2123+
Package::new("target-dep", "0.1.0").publish();
2124+
let p = project()
2125+
.file(
2126+
"Cargo.toml",
2127+
r#"
2128+
[package]
2129+
name = "foo"
2130+
version = "0.1.0"
2131+
edition = "2021"
2132+
2133+
[dependencies]
2134+
bar = { version = "0.1.0", optional = true }
2135+
2136+
[build-dependencies]
2137+
baz = { version = "0.1.0", optional = true }
2138+
2139+
[target.'cfg(target_os = "linux")'.dependencies]
2140+
target-dep = { version = "0.1.0", optional = true }
2141+
2142+
[features]
2143+
target-dep = ["dep:target-dep"]
2144+
"#,
2145+
)
2146+
.file("src/lib.rs", "")
2147+
.build();
2148+
2149+
p.cargo("fix --edition --allow-no-vcs")
2150+
.masquerade_as_nightly_cargo(&["edition2024"])
2151+
.with_stderr(
2152+
"\
2153+
[MIGRATING] Cargo.toml from 2021 edition to 2024
2154+
[FIXED] Cargo.toml (2 fixes)
2155+
[UPDATING] `dummy-registry` index
2156+
[LOCKING] 4 packages to latest compatible versions
2157+
[CHECKING] foo v0.1.0 ([CWD])
2158+
[MIGRATING] src/lib.rs from 2021 edition to 2024
2159+
[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [..]s
2160+
",
2161+
)
2162+
.run();
2163+
assert_eq!(
2164+
p.read_file("Cargo.toml"),
2165+
r#"
2166+
[package]
2167+
name = "foo"
2168+
version = "0.1.0"
2169+
edition = "2021"
2170+
2171+
[dependencies]
2172+
bar = { version = "0.1.0", optional = true }
2173+
2174+
[build-dependencies]
2175+
baz = { version = "0.1.0", optional = true }
2176+
2177+
[target.'cfg(target_os = "linux")'.dependencies]
2178+
target-dep = { version = "0.1.0", optional = true }
2179+
2180+
[features]
2181+
target-dep = ["dep:target-dep"]
2182+
bar = ["dep:bar"]
2183+
baz = ["dep:baz"]
2184+
"#
2185+
);
2186+
}

0 commit comments

Comments
 (0)