mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 14:13:58 +00:00
Make IncorrectDiagnostic match rustc by copying rustc's code.
This commit is contained in:
parent
d5c294f78c
commit
afb2166aeb
1 changed files with 122 additions and 125 deletions
|
@ -1,156 +1,145 @@
|
||||||
//! Functions for string case manipulation, such as detecting the identifier case,
|
//! Functions for string case manipulation, such as detecting the identifier case,
|
||||||
//! and converting it into appropriate form.
|
//! and converting it into appropriate form.
|
||||||
|
|
||||||
#[derive(Debug)]
|
// Code that was taken from rustc was taken at commit 89fdb30,
|
||||||
enum DetectedCase {
|
// from file /compiler/rustc_lint/src/nonstandard_style.rs
|
||||||
LowerCamelCase,
|
|
||||||
UpperCamelCase,
|
|
||||||
LowerSnakeCase,
|
|
||||||
UpperSnakeCase,
|
|
||||||
Unknown,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn detect_case(ident: &str) -> DetectedCase {
|
|
||||||
let trimmed_ident = ident.trim_matches('_');
|
|
||||||
let first_lowercase = trimmed_ident.starts_with(|chr: char| chr.is_ascii_lowercase());
|
|
||||||
let mut has_lowercase = first_lowercase;
|
|
||||||
let mut has_uppercase = false;
|
|
||||||
let mut has_underscore = false;
|
|
||||||
|
|
||||||
for chr in trimmed_ident.chars() {
|
|
||||||
if chr == '_' {
|
|
||||||
has_underscore = true;
|
|
||||||
} else if chr.is_ascii_uppercase() {
|
|
||||||
has_uppercase = true;
|
|
||||||
} else if chr.is_ascii_lowercase() {
|
|
||||||
has_lowercase = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if has_uppercase {
|
|
||||||
if !has_lowercase {
|
|
||||||
if has_underscore {
|
|
||||||
DetectedCase::UpperSnakeCase
|
|
||||||
} else {
|
|
||||||
// It has uppercase only and no underscores. Ex: "AABB"
|
|
||||||
// This is a camel cased acronym.
|
|
||||||
DetectedCase::UpperCamelCase
|
|
||||||
}
|
|
||||||
} else if !has_underscore {
|
|
||||||
if first_lowercase {
|
|
||||||
DetectedCase::LowerCamelCase
|
|
||||||
} else {
|
|
||||||
DetectedCase::UpperCamelCase
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// It has uppercase, it has lowercase, it has underscore.
|
|
||||||
// No assumptions here
|
|
||||||
DetectedCase::Unknown
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
DetectedCase::LowerSnakeCase
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts an identifier to an UpperCamelCase form.
|
/// Converts an identifier to an UpperCamelCase form.
|
||||||
/// Returns `None` if the string is already is UpperCamelCase.
|
/// Returns `None` if the string is already is UpperCamelCase.
|
||||||
pub fn to_camel_case(ident: &str) -> Option<String> {
|
pub fn to_camel_case(ident: &str) -> Option<String> {
|
||||||
let detected_case = detect_case(ident);
|
if is_camel_case(ident) {
|
||||||
|
return None;
|
||||||
match detected_case {
|
|
||||||
DetectedCase::UpperCamelCase => return None,
|
|
||||||
DetectedCase::LowerCamelCase => {
|
|
||||||
let mut first_capitalized = false;
|
|
||||||
let output = ident
|
|
||||||
.chars()
|
|
||||||
.map(|chr| {
|
|
||||||
if !first_capitalized && chr.is_ascii_lowercase() {
|
|
||||||
first_capitalized = true;
|
|
||||||
chr.to_ascii_uppercase()
|
|
||||||
} else {
|
|
||||||
chr
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
return Some(output);
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut output = String::with_capacity(ident.len());
|
// Taken from rustc.
|
||||||
|
let ret = ident
|
||||||
|
.trim_matches('_')
|
||||||
|
.split('_')
|
||||||
|
.filter(|component| !component.is_empty())
|
||||||
|
.map(|component| {
|
||||||
|
let mut camel_cased_component = String::new();
|
||||||
|
|
||||||
let mut capital_added = false;
|
let mut new_word = true;
|
||||||
for chr in ident.chars() {
|
let mut prev_is_lower_case = true;
|
||||||
if chr.is_alphabetic() {
|
|
||||||
if !capital_added {
|
for c in component.chars() {
|
||||||
output.push(chr.to_ascii_uppercase());
|
// Preserve the case if an uppercase letter follows a lowercase letter, so that
|
||||||
capital_added = true;
|
// `camelCase` is converted to `CamelCase`.
|
||||||
} else {
|
if prev_is_lower_case && c.is_uppercase() {
|
||||||
output.push(chr.to_ascii_lowercase());
|
new_word = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if new_word {
|
||||||
|
camel_cased_component.push_str(&c.to_uppercase().to_string());
|
||||||
|
} else {
|
||||||
|
camel_cased_component.push_str(&c.to_lowercase().to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
prev_is_lower_case = c.is_lowercase();
|
||||||
|
new_word = false;
|
||||||
}
|
}
|
||||||
} else if chr == '_' {
|
|
||||||
// Skip this character and make the next one capital.
|
|
||||||
capital_added = false;
|
|
||||||
} else {
|
|
||||||
// Put the characted as-is.
|
|
||||||
output.push(chr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if output == ident {
|
camel_cased_component
|
||||||
// While we didn't detect the correct case at the beginning, there
|
})
|
||||||
// may be special cases: e.g. `A` is both valid CamelCase and UPPER_SNAKE_CASE.
|
.fold((String::new(), None), |(acc, prev): (String, Option<String>), next| {
|
||||||
None
|
// separate two components with an underscore if their boundary cannot
|
||||||
} else {
|
// be distinguished using a uppercase/lowercase case distinction
|
||||||
Some(output)
|
let join = if let Some(prev) = prev {
|
||||||
}
|
let l = prev.chars().last().unwrap();
|
||||||
|
let f = next.chars().next().unwrap();
|
||||||
|
!char_has_case(l) && !char_has_case(f)
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
};
|
||||||
|
(acc + if join { "_" } else { "" } + &next, Some(next))
|
||||||
|
})
|
||||||
|
.0;
|
||||||
|
Some(ret)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Converts an identifier to a lower_snake_case form.
|
/// Converts an identifier to a lower_snake_case form.
|
||||||
/// Returns `None` if the string is already in lower_snake_case.
|
/// Returns `None` if the string is already in lower_snake_case.
|
||||||
pub fn to_lower_snake_case(ident: &str) -> Option<String> {
|
pub fn to_lower_snake_case(ident: &str) -> Option<String> {
|
||||||
// First, assume that it's UPPER_SNAKE_CASE.
|
if is_lower_snake_case(ident) {
|
||||||
match detect_case(ident) {
|
return None;
|
||||||
DetectedCase::LowerSnakeCase => return None,
|
} else if is_upper_snake_case(ident) {
|
||||||
DetectedCase::UpperSnakeCase => {
|
return Some(ident.to_lowercase());
|
||||||
return Some(ident.chars().map(|chr| chr.to_ascii_lowercase()).collect())
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Otherwise, assume that it's CamelCase.
|
Some(stdx::to_lower_snake_case(ident))
|
||||||
let lower_snake_case = stdx::to_lower_snake_case(ident);
|
|
||||||
|
|
||||||
if lower_snake_case == ident {
|
|
||||||
// While we didn't detect the correct case at the beginning, there
|
|
||||||
// may be special cases: e.g. `a` is both valid camelCase and snake_case.
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
Some(lower_snake_case)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Converts an identifier to an UPPER_SNAKE_CASE form.
|
/// Converts an identifier to an UPPER_SNAKE_CASE form.
|
||||||
/// Returns `None` if the string is already is UPPER_SNAKE_CASE.
|
/// Returns `None` if the string is already is UPPER_SNAKE_CASE.
|
||||||
pub fn to_upper_snake_case(ident: &str) -> Option<String> {
|
pub fn to_upper_snake_case(ident: &str) -> Option<String> {
|
||||||
match detect_case(ident) {
|
if is_upper_snake_case(ident) {
|
||||||
DetectedCase::UpperSnakeCase => return None,
|
return None;
|
||||||
DetectedCase::LowerSnakeCase => {
|
} else if is_lower_snake_case(ident) {
|
||||||
return Some(ident.chars().map(|chr| chr.to_ascii_uppercase()).collect())
|
return Some(ident.to_uppercase());
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Normalize the string from whatever form it's in currently, and then just make it uppercase.
|
Some(stdx::to_upper_snake_case(ident))
|
||||||
let upper_snake_case = stdx::to_upper_snake_case(ident);
|
}
|
||||||
|
|
||||||
if upper_snake_case == ident {
|
// Taken from rustc.
|
||||||
// While we didn't detect the correct case at the beginning, there
|
// Modified by replacing the use of unstable feature `array_windows`.
|
||||||
// may be special cases: e.g. `A` is both valid CamelCase and UPPER_SNAKE_CASE.
|
fn is_camel_case(name: &str) -> bool {
|
||||||
None
|
let name = name.trim_matches('_');
|
||||||
} else {
|
if name.is_empty() {
|
||||||
Some(upper_snake_case)
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut fst = None;
|
||||||
|
// start with a non-lowercase letter rather than non-uppercase
|
||||||
|
// ones (some scripts don't have a concept of upper/lowercase)
|
||||||
|
!name.chars().next().unwrap().is_lowercase()
|
||||||
|
&& !name.contains("__")
|
||||||
|
&& !name.chars().any(|snd| {
|
||||||
|
let ret = match (fst, snd) {
|
||||||
|
(None, _) => false,
|
||||||
|
(Some(fst), snd) => {
|
||||||
|
char_has_case(fst) && snd == '_' || char_has_case(snd) && fst == '_'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
fst = Some(snd);
|
||||||
|
|
||||||
|
ret
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_lower_snake_case(ident: &str) -> bool {
|
||||||
|
is_snake_case(ident, char::is_uppercase)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_upper_snake_case(ident: &str) -> bool {
|
||||||
|
is_snake_case(ident, char::is_lowercase)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Taken from rustc.
|
||||||
|
// Modified to allow checking for both upper and lower snake case.
|
||||||
|
fn is_snake_case<F: Fn(char) -> bool>(ident: &str, wrong_case: F) -> bool {
|
||||||
|
if ident.is_empty() {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
let ident = ident.trim_matches('_');
|
||||||
|
|
||||||
|
let mut allow_underscore = true;
|
||||||
|
ident.chars().all(|c| {
|
||||||
|
allow_underscore = match c {
|
||||||
|
'_' if !allow_underscore => return false,
|
||||||
|
'_' => false,
|
||||||
|
// It would be more obvious to check for the correct case,
|
||||||
|
// but some characters do not have a case.
|
||||||
|
c if !wrong_case(c) => true,
|
||||||
|
_ => return false,
|
||||||
|
};
|
||||||
|
true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// Taken from rustc.
|
||||||
|
fn char_has_case(c: char) -> bool {
|
||||||
|
c.is_lowercase() || c.is_uppercase()
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
@ -173,6 +162,7 @@ mod tests {
|
||||||
check(to_lower_snake_case, "CamelCase", expect![["camel_case"]]);
|
check(to_lower_snake_case, "CamelCase", expect![["camel_case"]]);
|
||||||
check(to_lower_snake_case, "lowerCamelCase", expect![["lower_camel_case"]]);
|
check(to_lower_snake_case, "lowerCamelCase", expect![["lower_camel_case"]]);
|
||||||
check(to_lower_snake_case, "a", expect![[""]]);
|
check(to_lower_snake_case, "a", expect![[""]]);
|
||||||
|
check(to_lower_snake_case, "abc", expect![[""]]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -187,6 +177,11 @@ mod tests {
|
||||||
check(to_camel_case, "name", expect![["Name"]]);
|
check(to_camel_case, "name", expect![["Name"]]);
|
||||||
check(to_camel_case, "A", expect![[""]]);
|
check(to_camel_case, "A", expect![[""]]);
|
||||||
check(to_camel_case, "AABB", expect![[""]]);
|
check(to_camel_case, "AABB", expect![[""]]);
|
||||||
|
// Taken from rustc: /compiler/rustc_lint/src/nonstandard_style/tests.rs
|
||||||
|
check(to_camel_case, "X86_64", expect![[""]]);
|
||||||
|
check(to_camel_case, "x86__64", expect![["X86_64"]]);
|
||||||
|
check(to_camel_case, "Abc_123", expect![["Abc123"]]);
|
||||||
|
check(to_camel_case, "A1_b2_c3", expect![["A1B2C3"]]);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -197,5 +192,7 @@ mod tests {
|
||||||
check(to_upper_snake_case, "CamelCase", expect![["CAMEL_CASE"]]);
|
check(to_upper_snake_case, "CamelCase", expect![["CAMEL_CASE"]]);
|
||||||
check(to_upper_snake_case, "lowerCamelCase", expect![["LOWER_CAMEL_CASE"]]);
|
check(to_upper_snake_case, "lowerCamelCase", expect![["LOWER_CAMEL_CASE"]]);
|
||||||
check(to_upper_snake_case, "A", expect![[""]]);
|
check(to_upper_snake_case, "A", expect![[""]]);
|
||||||
|
check(to_upper_snake_case, "ABC", expect![[""]]);
|
||||||
|
check(to_upper_snake_case, "X86_64", expect![[""]]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue