mirror of
https://github.com/rust-lang/rust-clippy
synced 2025-01-02 00:09:08 +00:00
55 lines
2 KiB
Rust
55 lines
2 KiB
Rust
|
//! lint on missing cargo common metadata
|
||
|
|
||
|
use cargo_metadata::Metadata;
|
||
|
use clippy_utils::diagnostics::span_lint;
|
||
|
use rustc_lint::LateContext;
|
||
|
use rustc_span::source_map::DUMMY_SP;
|
||
|
|
||
|
use super::CARGO_COMMON_METADATA;
|
||
|
|
||
|
pub(super) fn check(cx: &LateContext<'_>, metadata: &Metadata, ignore_publish: bool) {
|
||
|
for package in &metadata.packages {
|
||
|
// only run the lint if publish is `None` (`publish = true` or skipped entirely)
|
||
|
// or if the vector isn't empty (`publish = ["something"]`)
|
||
|
if package.publish.as_ref().filter(|publish| publish.is_empty()).is_none() || ignore_publish {
|
||
|
if is_empty_str(&package.description) {
|
||
|
missing_warning(cx, package, "package.description");
|
||
|
}
|
||
|
|
||
|
if is_empty_str(&package.license) && is_empty_str(&package.license_file) {
|
||
|
missing_warning(cx, package, "either package.license or package.license_file");
|
||
|
}
|
||
|
|
||
|
if is_empty_str(&package.repository) {
|
||
|
missing_warning(cx, package, "package.repository");
|
||
|
}
|
||
|
|
||
|
if is_empty_str(&package.readme) {
|
||
|
missing_warning(cx, package, "package.readme");
|
||
|
}
|
||
|
|
||
|
if is_empty_vec(&package.keywords) {
|
||
|
missing_warning(cx, package, "package.keywords");
|
||
|
}
|
||
|
|
||
|
if is_empty_vec(&package.categories) {
|
||
|
missing_warning(cx, package, "package.categories");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fn missing_warning(cx: &LateContext<'_>, package: &cargo_metadata::Package, field: &str) {
|
||
|
let message = format!("package `{}` is missing `{}` metadata", package.name, field);
|
||
|
span_lint(cx, CARGO_COMMON_METADATA, DUMMY_SP, &message);
|
||
|
}
|
||
|
|
||
|
fn is_empty_str<T: AsRef<std::ffi::OsStr>>(value: &Option<T>) -> bool {
|
||
|
value.as_ref().map_or(true, |s| s.as_ref().is_empty())
|
||
|
}
|
||
|
|
||
|
fn is_empty_vec(value: &[String]) -> bool {
|
||
|
// This works because empty iterators return true
|
||
|
value.iter().all(String::is_empty)
|
||
|
}
|