Skip to main content

cargo/lints/rules/
non_kebab_case_features.rs

1use std::path::Path;
2
3use annotate_snippets::AnnotationKind;
4use annotate_snippets::Group;
5use annotate_snippets::Level;
6use annotate_snippets::Origin;
7use annotate_snippets::Patch;
8use annotate_snippets::Snippet;
9use cargo_util_schemas::manifest::TomlToolLints;
10
11use crate::CargoResult;
12use crate::GlobalContext;
13use crate::core::Package;
14use crate::lints::Lint;
15use crate::lints::LintLevel;
16use crate::lints::LintLevelReason;
17use crate::lints::RESTRICTION;
18use crate::lints::get_key_value_span;
19use crate::lints::rel_cwd_manifest_path;
20
21pub static LINT: &Lint = &Lint {
22    name: "non_kebab_case_features",
23    desc: "features should have a kebab-case name",
24    primary_group: &RESTRICTION,
25    msrv: None,
26    edition_lint_opts: None,
27    feature_gate: None,
28    docs: Some(
29        r#"
30### What it does
31
32Detect feature names that are not kebab-case.
33
34### Why it is bad
35
36Having multiple naming styles within a workspace can be confusing.
37
38### Drawbacks
39
40Users would expect that a feature tightly coupled to a dependency would match the dependency's name.
41
42### Example
43
44```toml
45[features]
46foo_bar = []
47```
48
49Should be written as:
50
51```toml
52[features]
53foo-bar = []
54```
55"#,
56    ),
57};
58
59pub fn non_kebab_case_features(
60    pkg: &Package,
61    manifest_path: &Path,
62    cargo_lints: &TomlToolLints,
63    error_count: &mut usize,
64    gctx: &GlobalContext,
65) -> CargoResult<()> {
66    let (lint_level, reason) = LINT.level(
67        cargo_lints,
68        pkg.rust_version(),
69        pkg.manifest().edition(),
70        pkg.manifest().unstable_features(),
71    );
72
73    if lint_level == LintLevel::Allow {
74        return Ok(());
75    }
76
77    let manifest_path = rel_cwd_manifest_path(manifest_path, gctx);
78
79    lint_package(pkg, &manifest_path, lint_level, reason, error_count, gctx)
80}
81
82pub fn lint_package(
83    pkg: &Package,
84    manifest_path: &str,
85    lint_level: LintLevel,
86    reason: LintLevelReason,
87    error_count: &mut usize,
88    gctx: &GlobalContext,
89) -> CargoResult<()> {
90    for original_name in pkg.summary().features().keys() {
91        let original_name = &**original_name;
92        let kebab_case = heck::ToKebabCase::to_kebab_case(original_name);
93        if kebab_case == original_name {
94            continue;
95        }
96
97        let manifest = pkg.manifest();
98        let document = manifest.document();
99        let contents = manifest.contents();
100        let level = lint_level.to_diagnostic_level();
101        let emitted_source = LINT.emitted_source(lint_level, reason);
102
103        let mut primary = Group::with_title(level.primary_title(LINT.desc));
104        if let Some(document) = document
105            && let Some(contents) = contents
106            && let Some(span) = get_key_value_span(document, &["features", original_name])
107        {
108            primary = primary.element(
109                Snippet::source(contents)
110                    .path(manifest_path)
111                    .annotation(AnnotationKind::Primary.span(span.key)),
112            );
113        } else if let Some(document) = document
114            && let Some(contents) = contents
115            && let Some(dep_span) = get_key_value_span(document, &["dependencies", original_name])
116            && let Some(optional_span) =
117                get_key_value_span(document, &["dependencies", original_name, "optional"])
118        {
119            primary = primary.element(
120                Snippet::source(contents)
121                    .path(manifest_path)
122                    .annotation(AnnotationKind::Primary.span(dep_span.key).label("source of feature name"))
123                    .annotation(
124                        AnnotationKind::Context
125                            .span(optional_span.key.start..optional_span.value.end)
126                            .label("cause of feature"),
127                    ),
128            ).element(Level::NOTE.message("see also <https://doc.rust-lang.org/cargo/reference/features.html#optional-dependencies>"));
129        } else {
130            primary = primary.element(Origin::path(manifest_path));
131        }
132        primary = primary.element(Level::NOTE.message(emitted_source));
133        let mut report = vec![primary];
134        if let Some(document) = document
135            && let Some(contents) = contents
136            && let Some(span) = get_key_value_span(document, &["features", original_name])
137        {
138            let mut help = Group::with_title(Level::HELP.secondary_title(
139                "to change the feature name to kebab case, convert the `features` key",
140            ));
141            help = help.element(
142                Snippet::source(contents)
143                    .path(manifest_path)
144                    .patch(Patch::new(span.key, kebab_case.as_str())),
145            );
146            report.push(help);
147        }
148
149        if lint_level.is_error() {
150            *error_count += 1;
151        }
152        gctx.shell().print_report(&report, lint_level.force())?;
153    }
154
155    Ok(())
156}