From 3bfa3e8123dd2d652019ad270622025d10b87cc8 Mon Sep 17 00:00:00 2001 From: Benjamin Coenen <5719034+bnjjj@users.noreply.github.com> Date: Sun, 4 Oct 2020 21:21:30 +0200 Subject: [PATCH 01/22] when generating new function, focus on return type instead of body Signed-off-by: Benjamin Coenen <5719034+bnjjj@users.noreply.github.com> --- .../assists/src/handlers/generate_function.rs | 132 +++++++++--------- crates/assists/src/tests/generated.rs | 4 +- crates/syntax/src/ast/make.rs | 12 +- 3 files changed, 80 insertions(+), 68 deletions(-) diff --git a/crates/assists/src/handlers/generate_function.rs b/crates/assists/src/handlers/generate_function.rs index b38d640581..d23f4293b3 100644 --- a/crates/assists/src/handlers/generate_function.rs +++ b/crates/assists/src/handlers/generate_function.rs @@ -36,8 +36,8 @@ use crate::{ // bar("", baz()); // } // -// fn bar(arg: &str, baz: Baz) { -// ${0:todo!()} +// fn bar(arg: &str, baz: Baz) ${0:-> ()} { +// todo!() // } // // ``` @@ -80,9 +80,9 @@ pub(crate) fn generate_function(acc: &mut Assists, ctx: &AssistContext) -> Optio struct FunctionTemplate { insert_offset: TextSize, - placeholder_expr: ast::MacroCall, leading_ws: String, fn_def: ast::Fn, + ret_type: ast::RetType, trailing_ws: String, file: FileId, } @@ -90,11 +90,9 @@ struct FunctionTemplate { impl FunctionTemplate { fn to_string(&self, cap: Option) -> String { let f = match cap { - Some(cap) => render_snippet( - cap, - self.fn_def.syntax(), - Cursor::Replace(self.placeholder_expr.syntax()), - ), + Some(cap) => { + render_snippet(cap, self.fn_def.syntax(), Cursor::Replace(self.ret_type.syntax())) + } None => self.fn_def.to_string(), }; format!("{}{}{}", self.leading_ws, f, self.trailing_ws) @@ -141,8 +139,14 @@ impl FunctionBuilder { let placeholder_expr = make::expr_todo(); let fn_body = make::block_expr(vec![], Some(placeholder_expr)); let visibility = if self.needs_pub { Some(make::visibility_pub_crate()) } else { None }; - let mut fn_def = - make::fn_(visibility, self.fn_name, self.type_params, self.params, fn_body); + let mut fn_def = make::fn_( + visibility, + self.fn_name, + self.type_params, + self.params, + fn_body, + Some(make::ret_type(make::ty("()"))), + ); let leading_ws; let trailing_ws; @@ -163,12 +167,10 @@ impl FunctionBuilder { } }; - let placeholder_expr = - fn_def.syntax().descendants().find_map(ast::MacroCall::cast).unwrap(); FunctionTemplate { insert_offset, - placeholder_expr, leading_ws, + ret_type: fn_def.ret_type().unwrap(), fn_def, trailing_ws, file: self.file, @@ -349,8 +351,8 @@ fn foo() { bar(); } -fn bar() { - ${0:todo!()} +fn bar() ${0:-> ()} { + todo!() } ", ) @@ -376,8 +378,8 @@ impl Foo { } } -fn bar() { - ${0:todo!()} +fn bar() ${0:-> ()} { + todo!() } ", ) @@ -400,8 +402,8 @@ fn foo1() { bar(); } -fn bar() { - ${0:todo!()} +fn bar() ${0:-> ()} { + todo!() } fn foo2() {} @@ -426,8 +428,8 @@ mod baz { bar(); } - fn bar() { - ${0:todo!()} + fn bar() ${0:-> ()} { + todo!() } } ", @@ -452,8 +454,8 @@ fn foo() { bar(baz()); } -fn bar(baz: Baz) { - ${0:todo!()} +fn bar(baz: Baz) ${0:-> ()} { + todo!() } ", ); @@ -485,8 +487,8 @@ impl Baz { } } -fn bar(baz: Baz) { - ${0:todo!()} +fn bar(baz: Baz) ${0:-> ()} { + todo!() } ", ) @@ -506,8 +508,8 @@ fn foo() { bar("bar") } -fn bar(arg: &str) { - ${0:todo!()} +fn bar(arg: &str) ${0:-> ()} { + todo!() } "#, ) @@ -527,8 +529,8 @@ fn foo() { bar('x') } -fn bar(arg: char) { - ${0:todo!()} +fn bar(arg: char) ${0:-> ()} { + todo!() } "#, ) @@ -548,8 +550,8 @@ fn foo() { bar(42) } -fn bar(arg: i32) { - ${0:todo!()} +fn bar(arg: i32) ${0:-> ()} { + todo!() } ", ) @@ -569,8 +571,8 @@ fn foo() { bar(42 as u8) } -fn bar(arg: u8) { - ${0:todo!()} +fn bar(arg: u8) ${0:-> ()} { + todo!() } ", ) @@ -594,8 +596,8 @@ fn foo() { bar(x as u8) } -fn bar(x: u8) { - ${0:todo!()} +fn bar(x: u8) ${0:-> ()} { + todo!() } ", ) @@ -617,8 +619,8 @@ fn foo() { bar(worble) } -fn bar(worble: ()) { - ${0:todo!()} +fn bar(worble: ()) ${0:-> ()} { + todo!() } ", ) @@ -646,8 +648,8 @@ fn baz() { bar(foo()) } -fn bar(foo: impl Foo) { - ${0:todo!()} +fn bar(foo: impl Foo) ${0:-> ()} { + todo!() } ", ) @@ -673,8 +675,8 @@ fn foo() { bar(&baz()) } -fn bar(baz: &Baz) { - ${0:todo!()} +fn bar(baz: &Baz) ${0:-> ()} { + todo!() } ", ) @@ -702,8 +704,8 @@ fn foo() { bar(Baz::baz()) } -fn bar(baz: Baz::Bof) { - ${0:todo!()} +fn bar(baz: Baz::Bof) ${0:-> ()} { + todo!() } ", ) @@ -725,8 +727,8 @@ fn foo(t: T) { bar(t) } -fn bar(t: T) { - ${0:todo!()} +fn bar(t: T) ${0:-> ()} { + todo!() } ", ) @@ -756,8 +758,8 @@ fn foo() { bar(Baz::new); } -fn bar(arg: fn() -> Baz) { - ${0:todo!()} +fn bar(arg: fn() -> Baz) ${0:-> ()} { + todo!() } ", ) @@ -781,8 +783,8 @@ fn foo() { bar(closure) } -fn bar(closure: impl Fn(i64) -> i64) { - ${0:todo!()} +fn bar(closure: impl Fn(i64) -> i64) ${0:-> ()} { + todo!() } ", ) @@ -802,8 +804,8 @@ fn foo() { bar(baz) } -fn bar(baz: ()) { - ${0:todo!()} +fn bar(baz: ()) ${0:-> ()} { + todo!() } ", ) @@ -827,8 +829,8 @@ fn foo() { bar(baz(), baz()) } -fn bar(baz_1: Baz, baz_2: Baz) { - ${0:todo!()} +fn bar(baz_1: Baz, baz_2: Baz) ${0:-> ()} { + todo!() } ", ) @@ -852,8 +854,8 @@ fn foo() { bar(baz(), baz(), "foo", "bar") } -fn bar(baz_1: Baz, baz_2: Baz, arg_1: &str, arg_2: &str) { - ${0:todo!()} +fn bar(baz_1: Baz, baz_2: Baz, arg_1: &str, arg_2: &str) ${0:-> ()} { + todo!() } "#, ) @@ -872,8 +874,8 @@ fn foo() { ", r" mod bar { - pub(crate) fn my_fn() { - ${0:todo!()} + pub(crate) fn my_fn() ${0:-> ()} { + todo!() } } @@ -911,8 +913,8 @@ fn bar() { baz(foo) } -fn baz(foo: foo::Foo) { - ${0:todo!()} +fn baz(foo: foo::Foo) ${0:-> ()} { + todo!() } ", ) @@ -935,8 +937,8 @@ fn foo() { mod bar { fn something_else() {} - pub(crate) fn my_fn() { - ${0:todo!()} + pub(crate) fn my_fn() ${0:-> ()} { + todo!() } } @@ -963,8 +965,8 @@ fn foo() { r" mod bar { mod baz { - pub(crate) fn my_fn() { - ${0:todo!()} + pub(crate) fn my_fn() ${0:-> ()} { + todo!() } } } @@ -992,8 +994,8 @@ fn main() { r" -pub(crate) fn bar() { - ${0:todo!()} +pub(crate) fn bar() ${0:-> ()} { + todo!() }", ) } diff --git a/crates/assists/src/tests/generated.rs b/crates/assists/src/tests/generated.rs index 7f6e98a548..41f536574e 100644 --- a/crates/assists/src/tests/generated.rs +++ b/crates/assists/src/tests/generated.rs @@ -454,8 +454,8 @@ fn foo() { bar("", baz()); } -fn bar(arg: &str, baz: Baz) { - ${0:todo!()} +fn bar(arg: &str, baz: Baz) ${0:-> ()} { + todo!() } "#####, diff --git a/crates/syntax/src/ast/make.rs b/crates/syntax/src/ast/make.rs index 3a184094c8..74dbdfaf7b 100644 --- a/crates/syntax/src/ast/make.rs +++ b/crates/syntax/src/ast/make.rs @@ -320,6 +320,10 @@ pub fn param(name: String, ty: String) -> ast::Param { ast_from_text(&format!("fn f({}: {}) {{ }}", name, ty)) } +pub fn ret_type(ty: ast::Type) -> ast::RetType { + ast_from_text(&format!("fn f() -> {} {{ }}", ty)) +} + pub fn param_list(pats: impl IntoIterator) -> ast::ParamList { let args = pats.into_iter().join(", "); ast_from_text(&format!("fn f({}) {{ }}", args)) @@ -350,14 +354,20 @@ pub fn fn_( type_params: Option, params: ast::ParamList, body: ast::BlockExpr, + ret_type: Option, ) -> ast::Fn { let type_params = if let Some(type_params) = type_params { format!("<{}>", type_params) } else { "".into() }; + let ret_type = if let Some(ret_type) = ret_type { format!("{} ", ret_type) } else { "".into() }; let visibility = match visibility { None => String::new(), Some(it) => format!("{} ", it), }; - ast_from_text(&format!("{}fn {}{}{} {}", visibility, fn_name, type_params, params, body)) + + ast_from_text(&format!( + "{}fn {}{}{} {}{}", + visibility, fn_name, type_params, params, ret_type, body + )) } fn ast_from_text(text: &str) -> N { From 4039176ec63e5c75d76398f2debe26ac6fa59cbc Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 12:48:02 +0300 Subject: [PATCH 02/22] Create basic support for names case checks and implement function name case check --- crates/hir/src/code_model.rs | 31 ++++ crates/hir_def/src/data.rs | 2 + crates/hir_def/src/item_tree.rs | 2 + crates/hir_def/src/item_tree/lower.rs | 14 ++ crates/hir_ty/src/diagnostics.rs | 81 ++++++++- crates/hir_ty/src/diagnostics/decl_check.rs | 173 ++++++++++++++++++++ 6 files changed, 300 insertions(+), 3 deletions(-) create mode 100644 crates/hir_ty/src/diagnostics/decl_check.rs diff --git a/crates/hir/src/code_model.rs b/crates/hir/src/code_model.rs index 650b4fa40c..19ea26e36c 100644 --- a/crates/hir/src/code_model.rs +++ b/crates/hir/src/code_model.rs @@ -255,6 +255,37 @@ impl ModuleDef { ModuleDef::BuiltinType(it) => Some(it.as_name()), } } + + pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) { + match self { + ModuleDef::Adt(it) => match it { + Adt::Struct(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + Adt::Enum(it) => hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink), + Adt::Union(it) => hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink), + }, + ModuleDef::Trait(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + ModuleDef::Function(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + ModuleDef::TypeAlias(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + ModuleDef::Module(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + ModuleDef::Const(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + ModuleDef::Static(it) => { + hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) + } + _ => return, + } + } } pub use hir_def::{ diff --git a/crates/hir_def/src/data.rs b/crates/hir_def/src/data.rs index ff1ef0df64..733db2eaca 100644 --- a/crates/hir_def/src/data.rs +++ b/crates/hir_def/src/data.rs @@ -19,6 +19,7 @@ use crate::{ #[derive(Debug, Clone, PartialEq, Eq)] pub struct FunctionData { pub name: Name, + pub param_names: Vec>, pub params: Vec, pub ret_type: TypeRef, pub attrs: Attrs, @@ -39,6 +40,7 @@ impl FunctionData { Arc::new(FunctionData { name: func.name.clone(), + param_names: func.param_names.to_vec(), params: func.params.to_vec(), ret_type: func.ret_type.clone(), attrs: item_tree.attrs(ModItem::from(loc.id.value).into()).clone(), diff --git a/crates/hir_def/src/item_tree.rs b/crates/hir_def/src/item_tree.rs index 8a1121bbdf..ca502ce2ba 100644 --- a/crates/hir_def/src/item_tree.rs +++ b/crates/hir_def/src/item_tree.rs @@ -507,6 +507,8 @@ pub struct Function { pub has_self_param: bool, pub has_body: bool, pub is_unsafe: bool, + /// List of function parameters names. Does not include `self`. + pub param_names: Box<[Option]>, pub params: Box<[TypeRef]>, pub is_varargs: bool, pub ret_type: TypeRef, diff --git a/crates/hir_def/src/item_tree/lower.rs b/crates/hir_def/src/item_tree/lower.rs index 3328639cfe..2ffa46ac0e 100644 --- a/crates/hir_def/src/item_tree/lower.rs +++ b/crates/hir_def/src/item_tree/lower.rs @@ -283,6 +283,7 @@ impl Ctx { let name = func.name()?.as_name(); let mut params = Vec::new(); + let mut param_names = Vec::new(); let mut has_self_param = false; if let Some(param_list) = func.param_list() { if let Some(self_param) = param_list.self_param() { @@ -305,6 +306,18 @@ impl Ctx { has_self_param = true; } for param in param_list.params() { + let param_name = param + .pat() + .map(|name| { + if let ast::Pat::IdentPat(ident) = name { + Some(ident.name()?.as_name()) + } else { + None + } + }) + .flatten(); + param_names.push(param_name); + let type_ref = TypeRef::from_ast_opt(&self.body_ctx, param.ty()); params.push(type_ref); } @@ -341,6 +354,7 @@ impl Ctx { has_body, is_unsafe: func.unsafe_token().is_some(), params: params.into_boxed_slice(), + param_names: param_names.into_boxed_slice(), is_varargs, ret_type, ast_id, diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index 9ba005fabd..7227e7010f 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -2,10 +2,11 @@ mod expr; mod match_check; mod unsafe_check; +mod decl_check; -use std::any::Any; +use std::{any::Any, fmt}; -use hir_def::DefWithBodyId; +use hir_def::{DefWithBodyId, ModuleDefId}; use hir_expand::diagnostics::{Diagnostic, DiagnosticCode, DiagnosticSink}; use hir_expand::{name::Name, HirFileId, InFile}; use stdx::format_to; @@ -15,6 +16,16 @@ use crate::db::HirDatabase; pub use crate::diagnostics::expr::{record_literal_missing_fields, record_pattern_missing_fields}; +pub fn validate_module_item( + db: &dyn HirDatabase, + owner: ModuleDefId, + sink: &mut DiagnosticSink<'_>, +) { + let _p = profile::span("validate_body"); + let mut validator = decl_check::DeclValidator::new(owner, sink); + validator.validate_item(db); +} + pub fn validate_body(db: &dyn HirDatabase, owner: DefWithBodyId, sink: &mut DiagnosticSink<'_>) { let _p = profile::span("validate_body"); let infer = db.infer(owner); @@ -231,6 +242,64 @@ impl Diagnostic for MismatchedArgCount { } } +#[derive(Debug)] +pub enum CaseType { + // `some_var` + LowerSnakeCase, + // `SOME_CONST` + UpperSnakeCase, + // `SomeStruct` + UpperCamelCase, +} + +impl fmt::Display for CaseType { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + let repr = match self { + CaseType::LowerSnakeCase => "snake_case", + CaseType::UpperSnakeCase => "UPPER_SNAKE_CASE", + CaseType::UpperCamelCase => "UpperCamelCase", + }; + + write!(f, "{}", repr) + } +} + +#[derive(Debug)] +pub struct IncorrectCase { + pub file: HirFileId, + pub ident: SyntaxNodePtr, + pub expected_case: CaseType, + pub ident_text: String, + pub suggested_text: String, +} + +impl Diagnostic for IncorrectCase { + fn code(&self) -> DiagnosticCode { + DiagnosticCode("incorrect-ident-case") + } + + fn message(&self) -> String { + format!( + "Argument `{}` should have a {} name, e.g. `{}`", + self.ident_text, + self.expected_case.to_string(), + self.suggested_text + ) + } + + fn display_source(&self) -> InFile { + InFile::new(self.file, self.ident.clone()) + } + + fn as_any(&self) -> &(dyn Any + Send + 'static) { + self + } + + fn is_experimental(&self) -> bool { + true + } +} + #[cfg(test)] mod tests { use base_db::{fixture::WithFixture, FileId, SourceDatabase, SourceDatabaseExt}; @@ -242,7 +311,10 @@ mod tests { use rustc_hash::FxHashMap; use syntax::{TextRange, TextSize}; - use crate::{diagnostics::validate_body, test_db::TestDB}; + use crate::{ + diagnostics::{validate_body, validate_module_item}, + test_db::TestDB, + }; impl TestDB { fn diagnostics(&self, mut cb: F) { @@ -253,6 +325,9 @@ mod tests { let mut fns = Vec::new(); for (module_id, _) in crate_def_map.modules.iter() { for decl in crate_def_map[module_id].scope.declarations() { + let mut sink = DiagnosticSinkBuilder::new().build(&mut cb); + validate_module_item(self, decl, &mut sink); + if let ModuleDefId::FunctionId(f) = decl { fns.push(f) } diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs new file mode 100644 index 0000000000..6c3cd65c57 --- /dev/null +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -0,0 +1,173 @@ +//! Provides validators for the item declarations. +//! This includes the following items: +//! - variable bindings (e.g. `let x = foo();`) +//! - struct fields (e.g. `struct Foo { field: u8 }`) +//! - enum fields (e.g. `enum Foo { Variant { field: u8 } }`) +//! - function/method arguments (e.g. `fn foo(arg: u8)`) + +// TODO: Temporary, to not see warnings until module is somewhat complete. +// If you see these lines in the pull request, feel free to call me stupid :P. +#![allow(dead_code, unused_imports, unused_variables)] + +use std::sync::Arc; + +use hir_def::{ + body::Body, + db::DefDatabase, + expr::{Expr, ExprId, UnaryOp}, + item_tree::ItemTreeNode, + resolver::{resolver_for_expr, ResolveValueResult, ValueNs}, + src::HasSource, + AdtId, FunctionId, Lookup, ModuleDefId, +}; +use hir_expand::{diagnostics::DiagnosticSink, name::Name}; +use syntax::{ast::NameOwner, AstPtr}; + +use crate::{ + db::HirDatabase, + diagnostics::{CaseType, IncorrectCase}, + lower::CallableDefId, + ApplicationTy, InferenceResult, Ty, TypeCtor, +}; + +pub(super) struct DeclValidator<'a, 'b: 'a> { + owner: ModuleDefId, + sink: &'a mut DiagnosticSink<'b>, +} + +#[derive(Debug)] +struct Replacement { + current_name: Name, + suggested_text: String, + expected_case: CaseType, +} + +impl<'a, 'b> DeclValidator<'a, 'b> { + pub(super) fn new( + owner: ModuleDefId, + sink: &'a mut DiagnosticSink<'b>, + ) -> DeclValidator<'a, 'b> { + DeclValidator { owner, sink } + } + + pub(super) fn validate_item(&mut self, db: &dyn HirDatabase) { + // let def = self.owner.into(); + match self.owner { + ModuleDefId::FunctionId(func) => self.validate_func(db, func), + ModuleDefId::AdtId(adt) => self.validate_adt(db, adt), + _ => return, + } + } + + fn validate_func(&mut self, db: &dyn HirDatabase, func: FunctionId) { + let data = db.function_data(func); + + // 1. Check the function name. + let function_name = data.name.to_string(); + let fn_name_replacement = if let Some(new_name) = to_lower_snake_case(&function_name) { + let replacement = Replacement { + current_name: data.name.clone(), + suggested_text: new_name, + expected_case: CaseType::LowerSnakeCase, + }; + Some(replacement) + } else { + None + }; + + // 2. Check the param names. + let mut fn_param_replacements = Vec::new(); + + for param_name in data.param_names.iter().cloned().filter_map(|i| i) { + let name = param_name.to_string(); + if let Some(new_name) = to_lower_snake_case(&name) { + let replacement = Replacement { + current_name: param_name, + suggested_text: new_name, + expected_case: CaseType::LowerSnakeCase, + }; + fn_param_replacements.push(replacement); + } + } + + // 3. If there is at least one element to spawn a warning on, go to the source map and generate a warning. + self.create_incorrect_case_diagnostic_for_func( + func, + db, + fn_name_replacement, + fn_param_replacements, + ) + } + + /// Given the information about incorrect names in the function declaration, looks up into the source code + /// for exact locations and adds diagnostics into the sink. + fn create_incorrect_case_diagnostic_for_func( + &mut self, + func: FunctionId, + db: &dyn HirDatabase, + fn_name_replacement: Option, + fn_param_replacements: Vec, + ) { + // XXX: only look at sources if we do have incorrect names + if fn_name_replacement.is_none() && fn_param_replacements.is_empty() { + return; + } + + let fn_loc = func.lookup(db.upcast()); + let fn_src = fn_loc.source(db.upcast()); + + if let Some(replacement) = fn_name_replacement { + let ast_ptr = if let Some(name) = fn_src.value.name() { + name + } else { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement was generated for a function without a name: {:?}", + fn_src + ); + return; + }; + + let diagnostic = IncorrectCase { + file: fn_src.file_id, + ident: AstPtr::new(&ast_ptr).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } + + // let item_tree = db.item_tree(loc.id.file_id); + // let fn_def = &item_tree[fn_loc.id.value]; + // let (_, source_map) = db.body_with_source_map(func.into()); + } + + fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) {} +} + +fn to_lower_snake_case(ident: &str) -> Option { + let lower_snake_case = stdx::to_lower_snake_case(ident); + + if lower_snake_case == ident { + None + } else { + Some(lower_snake_case) + } +} + +#[cfg(test)] +mod tests { + use crate::diagnostics::tests::check_diagnostics; + + #[test] + fn incorrect_function_name() { + check_diagnostics( + r#" +fn NonSnakeCaseName() {} +// ^^^^^^^^^^^^^^^^ Argument `NonSnakeCaseName` should have a snake_case name, e.g. `non_snake_case_name` +"#, + ); + } +} From f5cea35986a0c8182ca427f10e20bc97ec564315 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 13:39:10 +0300 Subject: [PATCH 03/22] Add checks for function parameters --- crates/hir_ty/src/diagnostics.rs | 4 +- crates/hir_ty/src/diagnostics/decl_check.rs | 97 +++++++++++++++++++-- 2 files changed, 94 insertions(+), 7 deletions(-) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index 7227e7010f..24fff690a7 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -269,6 +269,7 @@ pub struct IncorrectCase { pub file: HirFileId, pub ident: SyntaxNodePtr, pub expected_case: CaseType, + pub ident_type: String, pub ident_text: String, pub suggested_text: String, } @@ -280,7 +281,8 @@ impl Diagnostic for IncorrectCase { fn message(&self) -> String { format!( - "Argument `{}` should have a {} name, e.g. `{}`", + "{} `{}` should have a {} name, e.g. `{}`", + self.ident_type, self.ident_text, self.expected_case.to_string(), self.suggested_text diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 6c3cd65c57..083df37726 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -21,7 +21,10 @@ use hir_def::{ AdtId, FunctionId, Lookup, ModuleDefId, }; use hir_expand::{diagnostics::DiagnosticSink, name::Name}; -use syntax::{ast::NameOwner, AstPtr}; +use syntax::{ + ast::{self, NameOwner}, + AstPtr, +}; use crate::{ db::HirDatabase, @@ -122,7 +125,8 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } else { // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. log::error!( - "Replacement was generated for a function without a name: {:?}", + "Replacement ({:?}) was generated for a function without a name: {:?}", + replacement, fn_src ); return; @@ -130,6 +134,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { let diagnostic = IncorrectCase { file: fn_src.file_id, + ident_type: "Function".to_string(), ident: AstPtr::new(&ast_ptr).into(), expected_case: replacement.expected_case, ident_text: replacement.current_name.to_string(), @@ -139,15 +144,71 @@ impl<'a, 'b> DeclValidator<'a, 'b> { self.sink.push(diagnostic); } - // let item_tree = db.item_tree(loc.id.file_id); - // let fn_def = &item_tree[fn_loc.id.value]; - // let (_, source_map) = db.body_with_source_map(func.into()); + let fn_params_list = match fn_src.value.param_list() { + Some(params) => params, + None => { + if !fn_param_replacements.is_empty() { + log::error!( + "Replacements ({:?}) were generated for a function parameters which had no parameters list: {:?}", + fn_param_replacements, fn_src + ); + } + return; + } + }; + let mut fn_params_iter = fn_params_list.params(); + for param_to_rename in fn_param_replacements { + // We assume that parameters in replacement are in the same order as in the + // actual params list, but just some of them (ones that named correctly) are skipped. + let ast_ptr = loop { + match fn_params_iter.next() { + Some(element) + if pat_equals_to_name(element.pat(), ¶m_to_rename.current_name) => + { + break element.pat().unwrap() + } + Some(_) => {} + None => { + log::error!( + "Replacement ({:?}) was generated for a function parameter which was not found: {:?}", + param_to_rename, fn_src + ); + return; + } + } + }; + + let diagnostic = IncorrectCase { + file: fn_src.file_id, + ident_type: "Argument".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: param_to_rename.expected_case, + ident_text: param_to_rename.current_name.to_string(), + suggested_text: param_to_rename.suggested_text, + }; + + self.sink.push(diagnostic); + } } fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) {} } +fn pat_equals_to_name(pat: Option, name: &Name) -> bool { + if let Some(ast::Pat::IdentPat(ident)) = pat { + ident.to_string() == name.to_string() + } else { + false + } +} + fn to_lower_snake_case(ident: &str) -> Option { + // First, assume that it's UPPER_SNAKE_CASE. + if let Some(normalized) = to_lower_snake_case_from_upper_snake_case(ident) { + return Some(normalized); + } + + // Otherwise, assume that it's CamelCase. let lower_snake_case = stdx::to_lower_snake_case(ident); if lower_snake_case == ident { @@ -157,6 +218,17 @@ fn to_lower_snake_case(ident: &str) -> Option { } } +fn to_lower_snake_case_from_upper_snake_case(ident: &str) -> Option { + let is_upper_snake_case = ident.chars().all(|c| c.is_ascii_uppercase() || c == '_'); + + if is_upper_snake_case { + let string = ident.chars().map(|c| c.to_ascii_lowercase()).collect(); + Some(string) + } else { + None + } +} + #[cfg(test)] mod tests { use crate::diagnostics::tests::check_diagnostics; @@ -166,7 +238,20 @@ mod tests { check_diagnostics( r#" fn NonSnakeCaseName() {} -// ^^^^^^^^^^^^^^^^ Argument `NonSnakeCaseName` should have a snake_case name, e.g. `non_snake_case_name` +// ^^^^^^^^^^^^^^^^ Function `NonSnakeCaseName` should have a snake_case name, e.g. `non_snake_case_name` +"#, + ); + } + + #[test] + fn incorrect_function_params() { + check_diagnostics( + r#" +fn foo(SomeParam: u8) {} + // ^^^^^^^^^ Argument `SomeParam` should have a snake_case name, e.g. `some_param` + +fn foo2(ok_param: &str, CAPS_PARAM: u8) {} + // ^^^^^^^^^^ Argument `CAPS_PARAM` should have a snake_case name, e.g. `caps_param` "#, ); } From 1773c6d154abe5da00b31bb16139addcaa443bbb Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 14:35:26 +0300 Subject: [PATCH 04/22] Extract helper functions into a separate module --- crates/hir_ty/src/diagnostics/decl_check.rs | 51 +++++----- .../src/diagnostics/decl_check/str_helpers.rs | 92 +++++++++++++++++++ 2 files changed, 113 insertions(+), 30 deletions(-) create mode 100644 crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 083df37726..1a0906492d 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -9,6 +9,8 @@ // If you see these lines in the pull request, feel free to call me stupid :P. #![allow(dead_code, unused_imports, unused_variables)] +mod str_helpers; + use std::sync::Arc; use hir_def::{ @@ -18,7 +20,7 @@ use hir_def::{ item_tree::ItemTreeNode, resolver::{resolver_for_expr, ResolveValueResult, ValueNs}, src::HasSource, - AdtId, FunctionId, Lookup, ModuleDefId, + AdtId, EnumId, FunctionId, Lookup, ModuleDefId, StructId, }; use hir_expand::{diagnostics::DiagnosticSink, name::Name}; use syntax::{ @@ -28,7 +30,7 @@ use syntax::{ use crate::{ db::HirDatabase, - diagnostics::{CaseType, IncorrectCase}, + diagnostics::{decl_check::str_helpers::*, CaseType, IncorrectCase}, lower::CallableDefId, ApplicationTy, InferenceResult, Ty, TypeCtor, }; @@ -191,7 +193,23 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } } - fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) {} + fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) { + match adt { + AdtId::StructId(struct_id) => self.validate_struct(db, struct_id), + AdtId::EnumId(enum_id) => self.validate_enum(db, enum_id), + AdtId::UnionId(_) => { + // Unions aren't yet supported by this validator. + } + } + } + + fn validate_struct(&mut self, db: &dyn HirDatabase, struct_id: StructId) { + let data = db.struct_data(struct_id); + } + + fn validate_enum(&mut self, db: &dyn HirDatabase, enum_id: EnumId) { + let data = db.enum_data(enum_id); + } } fn pat_equals_to_name(pat: Option, name: &Name) -> bool { @@ -202,33 +220,6 @@ fn pat_equals_to_name(pat: Option, name: &Name) -> bool { } } -fn to_lower_snake_case(ident: &str) -> Option { - // First, assume that it's UPPER_SNAKE_CASE. - if let Some(normalized) = to_lower_snake_case_from_upper_snake_case(ident) { - return Some(normalized); - } - - // Otherwise, assume that it's CamelCase. - let lower_snake_case = stdx::to_lower_snake_case(ident); - - if lower_snake_case == ident { - None - } else { - Some(lower_snake_case) - } -} - -fn to_lower_snake_case_from_upper_snake_case(ident: &str) -> Option { - let is_upper_snake_case = ident.chars().all(|c| c.is_ascii_uppercase() || c == '_'); - - if is_upper_snake_case { - let string = ident.chars().map(|c| c.to_ascii_lowercase()).collect(); - Some(string) - } else { - None - } -} - #[cfg(test)] mod tests { use crate::diagnostics::tests::check_diagnostics; diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs new file mode 100644 index 0000000000..3d8f1b5f2d --- /dev/null +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -0,0 +1,92 @@ +pub fn to_camel_case(ident: &str) -> Option { + let mut output = String::new(); + + if is_camel_case(ident) { + return None; + } + + let mut capital_added = false; + for chr in ident.chars() { + if chr.is_alphabetic() { + if !capital_added { + output.push(chr.to_ascii_uppercase()); + capital_added = true; + } else { + output.push(chr.to_ascii_lowercase()); + } + } 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 { + None + } else { + Some(output) + } +} + +pub fn to_lower_snake_case(ident: &str) -> Option { + // First, assume that it's UPPER_SNAKE_CASE. + if let Some(normalized) = to_lower_snake_case_from_upper_snake_case(ident) { + return Some(normalized); + } + + // Otherwise, assume that it's CamelCase. + let lower_snake_case = stdx::to_lower_snake_case(ident); + + if lower_snake_case == ident { + None + } else { + Some(lower_snake_case) + } +} + +fn to_lower_snake_case_from_upper_snake_case(ident: &str) -> Option { + if is_upper_snake_case(ident) { + let string = ident.chars().map(|c| c.to_ascii_lowercase()).collect(); + Some(string) + } else { + None + } +} + +fn is_upper_snake_case(ident: &str) -> bool { + ident.chars().all(|c| c.is_ascii_uppercase() || c == '_') +} + +fn is_camel_case(ident: &str) -> bool { + // We assume that the string is either snake case or camel case. + ident.chars().all(|c| c != '_') +} + +#[cfg(test)] +mod tests { + use super::*; + use expect_test::{expect, Expect}; + + fn check Option>(fun: F, input: &str, expect: Expect) { + // `None` is translated to empty string, meaning that there is nothing to fix. + let output = fun(input).unwrap_or_default(); + + expect.assert_eq(&output); + } + + #[test] + fn test_to_lower_snake_case() { + check(to_lower_snake_case, "lower_snake_case", expect![[""]]); + check(to_lower_snake_case, "UPPER_SNAKE_CASE", expect![["upper_snake_case"]]); + check(to_lower_snake_case, "CamelCase", expect![["camel_case"]]); + } + + #[test] + fn test_to_camel_case() { + check(to_camel_case, "CamelCase", expect![[""]]); + check(to_camel_case, "lower_snake_case", expect![["LowerSnakeCase"]]); + check(to_camel_case, "UPPER_SNAKE_CASE", expect![["UpperSnakeCase"]]); + } +} From 329626124f360feadb47e83be5690861c62a4b70 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 14:47:46 +0300 Subject: [PATCH 05/22] Add check for structure names to be CamelCase --- crates/hir_ty/src/diagnostics.rs | 2 +- crates/hir_ty/src/diagnostics/decl_check.rs | 138 ++++++++++++++++++++ 2 files changed, 139 insertions(+), 1 deletion(-) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index 24fff690a7..bd370e3b2d 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -257,7 +257,7 @@ impl fmt::Display for CaseType { let repr = match self { CaseType::LowerSnakeCase => "snake_case", CaseType::UpperSnakeCase => "UPPER_SNAKE_CASE", - CaseType::UpperCamelCase => "UpperCamelCase", + CaseType::UpperCamelCase => "CamelCase", }; write!(f, "{}", repr) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 1a0906492d..b7f511fd89 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -14,6 +14,7 @@ mod str_helpers; use std::sync::Arc; use hir_def::{ + adt::VariantData, body::Body, db::DefDatabase, expr::{Expr, ExprId, UnaryOp}, @@ -205,6 +206,133 @@ impl<'a, 'b> DeclValidator<'a, 'b> { fn validate_struct(&mut self, db: &dyn HirDatabase, struct_id: StructId) { let data = db.struct_data(struct_id); + + // 1. Check the structure name. + let struct_name = data.name.to_string(); + let struct_name_replacement = if let Some(new_name) = to_camel_case(&struct_name) { + let replacement = Replacement { + current_name: data.name.clone(), + suggested_text: new_name, + expected_case: CaseType::UpperCamelCase, + }; + Some(replacement) + } else { + None + }; + + // 2. Check the field names. + let mut struct_fields_replacements = Vec::new(); + + if let VariantData::Record(fields) = data.variant_data.as_ref() { + for (_, field) in fields.iter() { + let field_name = field.name.to_string(); + if let Some(new_name) = to_lower_snake_case(&field_name) { + let replacement = Replacement { + current_name: field.name.clone(), + suggested_text: new_name, + expected_case: CaseType::LowerSnakeCase, + }; + struct_fields_replacements.push(replacement); + } + } + } + + // 3. If there is at least one element to spawn a warning on, go to the source map and generate a warning. + self.create_incorrect_case_diagnostic_for_struct( + struct_id, + db, + struct_name_replacement, + struct_fields_replacements, + ) + } + + /// Given the information about incorrect names in the struct declaration, looks up into the source code + /// for exact locations and adds diagnostics into the sink. + fn create_incorrect_case_diagnostic_for_struct( + &mut self, + struct_id: StructId, + db: &dyn HirDatabase, + struct_name_replacement: Option, + struct_fields_replacements: Vec, + ) { + // XXX: only look at sources if we do have incorrect names + if struct_name_replacement.is_none() && struct_fields_replacements.is_empty() { + return; + } + + let struct_loc = struct_id.lookup(db.upcast()); + let struct_src = struct_loc.source(db.upcast()); + + if let Some(replacement) = struct_name_replacement { + let ast_ptr = if let Some(name) = struct_src.value.name() { + name + } else { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement ({:?}) was generated for a structure without a name: {:?}", + replacement, + struct_src + ); + return; + }; + + let diagnostic = IncorrectCase { + file: struct_src.file_id, + ident_type: "Structure".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } + + // let fn_params_list = match fn_src.value.param_list() { + // Some(params) => params, + // None => { + // if !fn_param_replacements.is_empty() { + // log::error!( + // "Replacements ({:?}) were generated for a function parameters which had no parameters list: {:?}", + // fn_param_replacements, fn_src + // ); + // } + // return; + // } + // }; + // let mut fn_params_iter = fn_params_list.params(); + // for param_to_rename in fn_param_replacements { + // // We assume that parameters in replacement are in the same order as in the + // // actual params list, but just some of them (ones that named correctly) are skipped. + // let ast_ptr = loop { + // match fn_params_iter.next() { + // Some(element) + // if pat_equals_to_name(element.pat(), ¶m_to_rename.current_name) => + // { + // break element.pat().unwrap() + // } + // Some(_) => {} + // None => { + // log::error!( + // "Replacement ({:?}) was generated for a function parameter which was not found: {:?}", + // param_to_rename, fn_src + // ); + // return; + // } + // } + // }; + + // let diagnostic = IncorrectCase { + // file: fn_src.file_id, + // ident_type: "Argument".to_string(), + // ident: AstPtr::new(&ast_ptr).into(), + // expected_case: param_to_rename.expected_case, + // ident_text: param_to_rename.current_name.to_string(), + // suggested_text: param_to_rename.suggested_text, + // }; + + // self.sink.push(diagnostic); + // } } fn validate_enum(&mut self, db: &dyn HirDatabase, enum_id: EnumId) { @@ -243,6 +371,16 @@ fn foo(SomeParam: u8) {} fn foo2(ok_param: &str, CAPS_PARAM: u8) {} // ^^^^^^^^^^ Argument `CAPS_PARAM` should have a snake_case name, e.g. `caps_param` +"#, + ); + } + + #[test] + fn incorrect_struct_name() { + check_diagnostics( + r#" +struct non_camel_case_name {} + // ^^^^^^^^^^^^^^^^^^^ Structure `non_camel_case_name` should have a CamelCase name, e.g. `NonCamelCaseName` "#, ); } From 21dd704b6b28374ea7bd2d1e13469be6807c4a8d Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 15:02:46 +0300 Subject: [PATCH 06/22] Check structure fields to be snake_case --- crates/hir_ty/src/diagnostics/decl_check.rs | 107 ++++++++++++-------- 1 file changed, 63 insertions(+), 44 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index b7f511fd89..260aa9607c 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -23,7 +23,10 @@ use hir_def::{ src::HasSource, AdtId, EnumId, FunctionId, Lookup, ModuleDefId, StructId, }; -use hir_expand::{diagnostics::DiagnosticSink, name::Name}; +use hir_expand::{ + diagnostics::DiagnosticSink, + name::{AsName, Name}, +}; use syntax::{ ast::{self, NameOwner}, AstPtr, @@ -288,51 +291,49 @@ impl<'a, 'b> DeclValidator<'a, 'b> { self.sink.push(diagnostic); } - // let fn_params_list = match fn_src.value.param_list() { - // Some(params) => params, - // None => { - // if !fn_param_replacements.is_empty() { - // log::error!( - // "Replacements ({:?}) were generated for a function parameters which had no parameters list: {:?}", - // fn_param_replacements, fn_src - // ); - // } - // return; - // } - // }; - // let mut fn_params_iter = fn_params_list.params(); - // for param_to_rename in fn_param_replacements { - // // We assume that parameters in replacement are in the same order as in the - // // actual params list, but just some of them (ones that named correctly) are skipped. - // let ast_ptr = loop { - // match fn_params_iter.next() { - // Some(element) - // if pat_equals_to_name(element.pat(), ¶m_to_rename.current_name) => - // { - // break element.pat().unwrap() - // } - // Some(_) => {} - // None => { - // log::error!( - // "Replacement ({:?}) was generated for a function parameter which was not found: {:?}", - // param_to_rename, fn_src - // ); - // return; - // } - // } - // }; + let struct_fields_list = match struct_src.value.field_list() { + Some(ast::FieldList::RecordFieldList(fields)) => fields, + _ => { + if !struct_fields_replacements.is_empty() { + log::error!( + "Replacements ({:?}) were generated for a structure fields which had no fields list: {:?}", + struct_fields_replacements, struct_src + ); + } + return; + } + }; + let mut struct_fields_iter = struct_fields_list.fields(); + for field_to_rename in struct_fields_replacements { + // We assume that parameters in replacement are in the same order as in the + // actual params list, but just some of them (ones that named correctly) are skipped. + let ast_ptr = loop { + match struct_fields_iter.next() { + Some(element) if names_equal(element.name(), &field_to_rename.current_name) => { + break element.name().unwrap() + } + Some(_) => {} + None => { + log::error!( + "Replacement ({:?}) was generated for a function parameter which was not found: {:?}", + field_to_rename, struct_src + ); + return; + } + } + }; - // let diagnostic = IncorrectCase { - // file: fn_src.file_id, - // ident_type: "Argument".to_string(), - // ident: AstPtr::new(&ast_ptr).into(), - // expected_case: param_to_rename.expected_case, - // ident_text: param_to_rename.current_name.to_string(), - // suggested_text: param_to_rename.suggested_text, - // }; + let diagnostic = IncorrectCase { + file: struct_src.file_id, + ident_type: "Field".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: field_to_rename.expected_case, + ident_text: field_to_rename.current_name.to_string(), + suggested_text: field_to_rename.suggested_text, + }; - // self.sink.push(diagnostic); - // } + self.sink.push(diagnostic); + } } fn validate_enum(&mut self, db: &dyn HirDatabase, enum_id: EnumId) { @@ -340,6 +341,14 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } } +fn names_equal(left: Option, right: &Name) -> bool { + if let Some(left) = left { + &left.as_name() == right + } else { + false + } +} + fn pat_equals_to_name(pat: Option, name: &Name) -> bool { if let Some(ast::Pat::IdentPat(ident)) = pat { ident.to_string() == name.to_string() @@ -381,6 +390,16 @@ fn foo2(ok_param: &str, CAPS_PARAM: u8) {} r#" struct non_camel_case_name {} // ^^^^^^^^^^^^^^^^^^^ Structure `non_camel_case_name` should have a CamelCase name, e.g. `NonCamelCaseName` +"#, + ); + } + + #[test] + fn incorrect_struct_field() { + check_diagnostics( + r#" +struct SomeStruct { SomeField: u8 } + // ^^^^^^^^^ Field `SomeField` should have a snake_case name, e.g. `some_field` "#, ); } From 17f1026c46e6e3797caf3c69737f66bd612c58e1 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 16:45:16 +0300 Subject: [PATCH 07/22] Improve string helpers functions --- crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs | 9 ++++++++- crates/stdx/src/lib.rs | 6 +++++- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs index 3d8f1b5f2d..953d0276fb 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -61,7 +61,9 @@ fn is_upper_snake_case(ident: &str) -> bool { fn is_camel_case(ident: &str) -> bool { // We assume that the string is either snake case or camel case. - ident.chars().all(|c| c != '_') + // `_` is allowed only at the beginning or in the end of identifier, not between characters. + ident.trim_matches('_').chars().all(|c| c != '_') + && ident.chars().find(|c| c.is_alphabetic()).map(|c| c.is_ascii_uppercase()).unwrap_or(true) } #[cfg(test)] @@ -80,13 +82,18 @@ mod tests { fn test_to_lower_snake_case() { check(to_lower_snake_case, "lower_snake_case", expect![[""]]); check(to_lower_snake_case, "UPPER_SNAKE_CASE", expect![["upper_snake_case"]]); + check(to_lower_snake_case, "Weird_Case", expect![["weird_case"]]); check(to_lower_snake_case, "CamelCase", expect![["camel_case"]]); } #[test] fn test_to_camel_case() { check(to_camel_case, "CamelCase", expect![[""]]); + check(to_camel_case, "CamelCase_", expect![[""]]); + check(to_camel_case, "_CamelCase", expect![[""]]); check(to_camel_case, "lower_snake_case", expect![["LowerSnakeCase"]]); check(to_camel_case, "UPPER_SNAKE_CASE", expect![["UpperSnakeCase"]]); + check(to_camel_case, "Weird_Case", expect![["WeirdCase"]]); + check(to_camel_case, "name", expect![["Name"]]); } } diff --git a/crates/stdx/src/lib.rs b/crates/stdx/src/lib.rs index 011935cade..522a9c1abd 100644 --- a/crates/stdx/src/lib.rs +++ b/crates/stdx/src/lib.rs @@ -32,8 +32,12 @@ pub fn to_lower_snake_case(s: &str) -> String { let mut buf = String::with_capacity(s.len()); let mut prev = false; for c in s.chars() { + // `&& prev` is required to not insert `_` before the first symbol. if c.is_ascii_uppercase() && prev { - buf.push('_') + // This check is required to not translate `Weird_Case` into `weird__case`. + if buf.chars().last() != Some('_') { + buf.push('_') + } } prev = true; From e24e22f288eba33928a9e579f13653d6f04fcdfa Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 17:34:52 +0300 Subject: [PATCH 08/22] Add fix for incorrect case diagnostic --- crates/hir/src/code_model.rs | 40 +++++++----------- crates/hir/src/diagnostics.rs | 3 +- crates/hir_ty/src/diagnostics.rs | 2 +- crates/ide/src/diagnostics.rs | 64 +++++++++++++++++++++++++++++ crates/ide/src/diagnostics/fixes.rs | 20 ++++++++- crates/ide/src/references.rs | 2 +- crates/ide/src/references/rename.rs | 7 ++++ crates/syntax/src/ptr.rs | 4 ++ 8 files changed, 112 insertions(+), 30 deletions(-) diff --git a/crates/hir/src/code_model.rs b/crates/hir/src/code_model.rs index 19ea26e36c..c134356ef1 100644 --- a/crates/hir/src/code_model.rs +++ b/crates/hir/src/code_model.rs @@ -257,34 +257,22 @@ impl ModuleDef { } pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) { - match self { + let id = match self { ModuleDef::Adt(it) => match it { - Adt::Struct(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - Adt::Enum(it) => hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink), - Adt::Union(it) => hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink), + Adt::Struct(it) => it.id.into(), + Adt::Enum(it) => it.id.into(), + Adt::Union(it) => it.id.into(), }, - ModuleDef::Trait(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - ModuleDef::Function(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - ModuleDef::TypeAlias(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - ModuleDef::Module(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - ModuleDef::Const(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } - ModuleDef::Static(it) => { - hir_ty::diagnostics::validate_module_item(db, it.id.into(), sink) - } + ModuleDef::Trait(it) => it.id.into(), + ModuleDef::Function(it) => it.id.into(), + ModuleDef::TypeAlias(it) => it.id.into(), + ModuleDef::Module(it) => it.id.into(), + ModuleDef::Const(it) => it.id.into(), + ModuleDef::Static(it) => it.id.into(), _ => return, - } + }; + + hir_ty::diagnostics::validate_module_item(db, id, sink) } } @@ -389,6 +377,8 @@ impl Module { let crate_def_map = db.crate_def_map(self.id.krate); crate_def_map.add_diagnostics(db.upcast(), self.id.local_id, sink); for decl in self.declarations(db) { + decl.diagnostics(db, sink); + match decl { crate::ModuleDef::Function(f) => f.diagnostics(db, sink), crate::ModuleDef::Module(m) => { diff --git a/crates/hir/src/diagnostics.rs b/crates/hir/src/diagnostics.rs index 363164b9b4..da2b40849a 100644 --- a/crates/hir/src/diagnostics.rs +++ b/crates/hir/src/diagnostics.rs @@ -2,5 +2,6 @@ pub use hir_def::diagnostics::UnresolvedModule; pub use hir_expand::diagnostics::{Diagnostic, DiagnosticSink, DiagnosticSinkBuilder}; pub use hir_ty::diagnostics::{ - MismatchedArgCount, MissingFields, MissingMatchArms, MissingOkInTailExpr, NoSuchField, + IncorrectCase, MismatchedArgCount, MissingFields, MissingMatchArms, MissingOkInTailExpr, + NoSuchField, }; diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index bd370e3b2d..66762b90e7 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -298,7 +298,7 @@ impl Diagnostic for IncorrectCase { } fn is_experimental(&self) -> bool { - true + false } } diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index f5d627b6ef..71ab98c1f0 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -96,6 +96,9 @@ pub(crate) fn diagnostics( .on::(|d| { res.borrow_mut().push(diagnostic_with_fix(d, &sema)); }) + .on::(|d| { + res.borrow_mut().push(warning_with_fix(d, &sema)); + }) // Only collect experimental diagnostics when they're enabled. .filter(|diag| !(diag.is_experimental() && config.disable_experimental)) .filter(|diag| !config.disabled.contains(diag.code().as_str())); @@ -130,6 +133,16 @@ fn diagnostic_with_fix(d: &D, sema: &Semantics(d: &D, sema: &Semantics) -> Diagnostic { + Diagnostic { + // name: Some(d.name().into()), + range: sema.diagnostics_display_range(d).range, + message: d.message(), + severity: Severity::WeakWarning, + fix: d.fix(&sema), + } +} + fn check_unnecessary_braces_in_use_statement( acc: &mut Vec, file_id: FileId, @@ -253,6 +266,37 @@ mod tests { ); } + /// Similar to `check_fix`, but applies all the available fixes. + fn check_fixes(ra_fixture_before: &str, ra_fixture_after: &str) { + let after = trim_indent(ra_fixture_after); + + let (analysis, file_position) = fixture::position(ra_fixture_before); + let diagnostic = analysis + .diagnostics(&DiagnosticsConfig::default(), file_position.file_id) + .unwrap() + .pop() + .unwrap(); + let fix = diagnostic.fix.unwrap(); + let target_file_contents = analysis.file_text(file_position.file_id).unwrap(); + let actual = { + let mut actual = target_file_contents.to_string(); + // Go from the last one to the first one, so that ranges won't be affected by previous edits. + for edit in fix.source_change.source_file_edits.iter().rev() { + edit.edit.apply(&mut actual); + } + actual + }; + + assert_eq_text!(&after, &actual); + assert!( + fix.fix_trigger_range.start() <= file_position.offset + && fix.fix_trigger_range.end() >= file_position.offset, + "diagnostic fix range {:?} does not touch cursor position {:?}", + fix.fix_trigger_range, + file_position.offset + ); + } + /// Checks that a diagnostic applies to the file containing the `<|>` cursor marker /// which has a fix that can apply to other files. fn check_apply_diagnostic_fix_in_other_file(ra_fixture_before: &str, ra_fixture_after: &str) { @@ -790,4 +834,24 @@ struct Foo { let diagnostics = analysis.diagnostics(&DiagnosticsConfig::default(), file_id).unwrap(); assert!(!diagnostics.is_empty()); } + + #[test] + fn test_rename_incorrect_case() { + check_fixes( + r#" +pub struct test_struct<|> { one: i32 } + +pub fn some_fn(val: test_struct) -> test_struct { + test_struct { one: val.one + 1 } +} +"#, + r#" +pub struct TestStruct { one: i32 } + +pub fn some_fn(val: TestStruct) -> TestStruct { + TestStruct { one: val.one + 1 } +} +"#, + ); + } } diff --git a/crates/ide/src/diagnostics/fixes.rs b/crates/ide/src/diagnostics/fixes.rs index 68ae1c2398..286ef07850 100644 --- a/crates/ide/src/diagnostics/fixes.rs +++ b/crates/ide/src/diagnostics/fixes.rs @@ -3,7 +3,10 @@ use base_db::FileId; use hir::{ db::AstDatabase, - diagnostics::{Diagnostic, MissingFields, MissingOkInTailExpr, NoSuchField, UnresolvedModule}, + diagnostics::{ + Diagnostic, IncorrectCase, MissingFields, MissingOkInTailExpr, NoSuchField, + UnresolvedModule, + }, HasSource, HirDisplay, Semantics, VariantDef, }; use ide_db::{ @@ -17,7 +20,7 @@ use syntax::{ }; use text_edit::TextEdit; -use crate::diagnostics::Fix; +use crate::{diagnostics::Fix, references::rename::rename_with_semantics, FilePosition}; /// A [Diagnostic] that potentially has a fix available. /// @@ -99,6 +102,19 @@ impl DiagnosticWithFix for MissingOkInTailExpr { } } +impl DiagnosticWithFix for IncorrectCase { + fn fix(&self, sema: &Semantics) -> Option { + let file_id = self.file.original_file(sema.db); + let offset = self.ident.text_range().start(); + let file_position = FilePosition { file_id, offset }; + + let rename_changes = rename_with_semantics(sema, file_position, &self.suggested_text)?; + + let label = format!("Rename to {}", self.suggested_text); + Some(Fix::new(&label, rename_changes.info, rename_changes.range)) + } +} + fn missing_record_expr_field_fix( sema: &Semantics, usage_file_id: FileId, diff --git a/crates/ide/src/references.rs b/crates/ide/src/references.rs index f65a05ea33..88e2f2db3f 100644 --- a/crates/ide/src/references.rs +++ b/crates/ide/src/references.rs @@ -9,7 +9,7 @@ //! at the index that the match starts at and its tree parent is //! resolved to the search element definition, we get a reference. -mod rename; +pub(crate) mod rename; use hir::Semantics; use ide_db::{ diff --git a/crates/ide/src/references/rename.rs b/crates/ide/src/references/rename.rs index f3b5cfc8c1..f9a11e43d8 100644 --- a/crates/ide/src/references/rename.rs +++ b/crates/ide/src/references/rename.rs @@ -42,7 +42,14 @@ pub(crate) fn rename( new_name: &str, ) -> Result, RenameError> { let sema = Semantics::new(db); + rename_with_semantics(&sema, position, new_name) +} +pub(crate) fn rename_with_semantics( + sema: &Semantics, + position: FilePosition, + new_name: &str, +) -> Result, RenameError> { match lex_single_syntax_kind(new_name) { Some(res) => match res { (SyntaxKind::IDENT, _) => (), diff --git a/crates/syntax/src/ptr.rs b/crates/syntax/src/ptr.rs index d3fb7a5d98..34e20464a7 100644 --- a/crates/syntax/src/ptr.rs +++ b/crates/syntax/src/ptr.rs @@ -23,6 +23,10 @@ impl SyntaxNodePtr { SyntaxNodePtr { range: node.text_range(), kind: node.kind() } } + pub fn text_range(&self) -> TextRange { + self.range.clone() + } + pub fn to_node(&self, root: &SyntaxNode) -> SyntaxNode { assert!(root.parent().is_none()); successors(Some(root.clone()), |node| { From fb96bba87895c062a78e6599cea161e461ff607d Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sat, 3 Oct 2020 18:01:25 +0300 Subject: [PATCH 09/22] Add diagnostics for enum names and variants --- crates/hir_ty/src/diagnostics.rs | 2 +- crates/hir_ty/src/diagnostics/decl_check.rs | 147 +++++++++++++++++++- crates/ide/src/diagnostics.rs | 26 ++++ 3 files changed, 173 insertions(+), 2 deletions(-) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index 66762b90e7..bd370e3b2d 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -298,7 +298,7 @@ impl Diagnostic for IncorrectCase { } fn is_experimental(&self) -> bool { - false + true } } diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 260aa9607c..7fc9c564e1 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -315,7 +315,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { Some(_) => {} None => { log::error!( - "Replacement ({:?}) was generated for a function parameter which was not found: {:?}", + "Replacement ({:?}) was generated for a structure field which was not found: {:?}", field_to_rename, struct_src ); return; @@ -338,6 +338,131 @@ impl<'a, 'b> DeclValidator<'a, 'b> { fn validate_enum(&mut self, db: &dyn HirDatabase, enum_id: EnumId) { let data = db.enum_data(enum_id); + + // 1. Check the enum name. + let enum_name = data.name.to_string(); + let enum_name_replacement = if let Some(new_name) = to_camel_case(&enum_name) { + let replacement = Replacement { + current_name: data.name.clone(), + suggested_text: new_name, + expected_case: CaseType::UpperCamelCase, + }; + Some(replacement) + } else { + None + }; + + // 2. Check the field names. + let mut enum_fields_replacements = Vec::new(); + + for (_, variant) in data.variants.iter() { + let variant_name = variant.name.to_string(); + if let Some(new_name) = to_camel_case(&variant_name) { + let replacement = Replacement { + current_name: variant.name.clone(), + suggested_text: new_name, + expected_case: CaseType::UpperCamelCase, + }; + enum_fields_replacements.push(replacement); + } + } + + // 3. If there is at least one element to spawn a warning on, go to the source map and generate a warning. + self.create_incorrect_case_diagnostic_for_enum( + enum_id, + db, + enum_name_replacement, + enum_fields_replacements, + ) + } + + /// Given the information about incorrect names in the struct declaration, looks up into the source code + /// for exact locations and adds diagnostics into the sink. + fn create_incorrect_case_diagnostic_for_enum( + &mut self, + enum_id: EnumId, + db: &dyn HirDatabase, + enum_name_replacement: Option, + enum_variants_replacements: Vec, + ) { + // XXX: only look at sources if we do have incorrect names + if enum_name_replacement.is_none() && enum_variants_replacements.is_empty() { + return; + } + + let enum_loc = enum_id.lookup(db.upcast()); + let enum_src = enum_loc.source(db.upcast()); + + if let Some(replacement) = enum_name_replacement { + let ast_ptr = if let Some(name) = enum_src.value.name() { + name + } else { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement ({:?}) was generated for a enum without a name: {:?}", + replacement, + enum_src + ); + return; + }; + + let diagnostic = IncorrectCase { + file: enum_src.file_id, + ident_type: "Enum".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } + + let enum_variants_list = match enum_src.value.variant_list() { + Some(variants) => variants, + _ => { + if !enum_variants_replacements.is_empty() { + log::error!( + "Replacements ({:?}) were generated for a enum variants which had no fields list: {:?}", + enum_variants_replacements, enum_src + ); + } + return; + } + }; + let mut enum_variants_iter = enum_variants_list.variants(); + for variant_to_rename in enum_variants_replacements { + // We assume that parameters in replacement are in the same order as in the + // actual params list, but just some of them (ones that named correctly) are skipped. + let ast_ptr = loop { + match enum_variants_iter.next() { + Some(variant) + if names_equal(variant.name(), &variant_to_rename.current_name) => + { + break variant.name().unwrap() + } + Some(_) => {} + None => { + log::error!( + "Replacement ({:?}) was generated for a enum variant which was not found: {:?}", + variant_to_rename, enum_src + ); + return; + } + } + }; + + let diagnostic = IncorrectCase { + file: enum_src.file_id, + ident_type: "Variant".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: variant_to_rename.expected_case, + ident_text: variant_to_rename.current_name.to_string(), + suggested_text: variant_to_rename.suggested_text, + }; + + self.sink.push(diagnostic); + } } } @@ -400,6 +525,26 @@ struct non_camel_case_name {} r#" struct SomeStruct { SomeField: u8 } // ^^^^^^^^^ Field `SomeField` should have a snake_case name, e.g. `some_field` +"#, + ); + } + + #[test] + fn incorrect_enum_name() { + check_diagnostics( + r#" +enum some_enum { Val(u8) } + // ^^^^^^^^^ Enum `some_enum` should have a CamelCase name, e.g. `SomeEnum` +"#, + ); + } + + #[test] + fn incorrect_enum_variant_name() { + check_diagnostics( + r#" +enum SomeEnum { SOME_VARIANT(u8) } + // ^^^^^^^^^^^^ Variant `SOME_VARIANT` should have a CamelCase name, e.g. `SomeVariant` "#, ); } diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index 71ab98c1f0..ad1b265fdf 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -851,6 +851,32 @@ pub struct TestStruct { one: i32 } pub fn some_fn(val: TestStruct) -> TestStruct { TestStruct { one: val.one + 1 } } +"#, + ); + + check_fixes( + r#" +pub fn some_fn(NonSnakeCase<|>: u8) -> u8 { + NonSnakeCase +} +"#, + r#" +pub fn some_fn(non_snake_case: u8) -> u8 { + non_snake_case +} +"#, + ); + + check_fixes( + r#" +pub fn SomeFn<|>(val: u8) -> u8 { + if val != 0 { SomeFn(val - 1) } else { val } +} +"#, + r#" +pub fn some_fn(val: u8) -> u8 { + if val != 0 { some_fn(val - 1) } else { val } +} "#, ); } From 9ec1741b651bd13e4e5e6224f2e2c5c503846a6b Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 07:37:43 +0300 Subject: [PATCH 10/22] Refactor string helpers for decl_check module --- .../src/diagnostics/decl_check/str_helpers.rs | 133 +++++++++++++----- 1 file changed, 99 insertions(+), 34 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs index 953d0276fb..e3826909bb 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -1,10 +1,74 @@ -pub fn to_camel_case(ident: &str) -> Option { - let mut output = String::new(); +#[derive(Debug)] +enum DetectedCase { + LowerCamelCase, + UpperCamelCase, + LowerSnakeCase, + UpperSnakeCase, + Unknown, +} - if is_camel_case(ident) { - return None; +fn detect_case(ident: &str) -> DetectedCase { + let trimmed_ident = ident.trim_matches('_'); + let first_lowercase = + trimmed_ident.chars().next().map(|chr| chr.is_ascii_lowercase()).unwrap_or(false); + 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 { + DetectedCase::UpperSnakeCase + } 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 + } +} + +pub fn to_camel_case(ident: &str) -> Option { + let detected_case = detect_case(ident); + + 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()); + let mut capital_added = false; for chr in ident.chars() { if chr.is_alphabetic() { @@ -23,47 +87,37 @@ pub fn to_camel_case(ident: &str) -> Option { } } - if output == ident { - None - } else { - Some(output) - } + Some(output) } pub fn to_lower_snake_case(ident: &str) -> Option { // First, assume that it's UPPER_SNAKE_CASE. - if let Some(normalized) = to_lower_snake_case_from_upper_snake_case(ident) { - return Some(normalized); + match detect_case(ident) { + DetectedCase::LowerSnakeCase => return None, + DetectedCase::UpperSnakeCase => { + return Some(ident.chars().map(|chr| chr.to_ascii_lowercase()).collect()) + } + _ => {} } // Otherwise, assume that it's CamelCase. let lower_snake_case = stdx::to_lower_snake_case(ident); + Some(lower_snake_case) +} - if lower_snake_case == ident { - None - } else { - Some(lower_snake_case) +pub fn to_upper_snake_case(ident: &str) -> Option { + match detect_case(ident) { + DetectedCase::UpperSnakeCase => return None, + DetectedCase::LowerSnakeCase => { + return Some(ident.chars().map(|chr| chr.to_ascii_uppercase()).collect()) + } + _ => {} } -} -fn to_lower_snake_case_from_upper_snake_case(ident: &str) -> Option { - if is_upper_snake_case(ident) { - let string = ident.chars().map(|c| c.to_ascii_lowercase()).collect(); - Some(string) - } else { - None - } -} - -fn is_upper_snake_case(ident: &str) -> bool { - ident.chars().all(|c| c.is_ascii_uppercase() || c == '_') -} - -fn is_camel_case(ident: &str) -> bool { - // We assume that the string is either snake case or camel case. - // `_` is allowed only at the beginning or in the end of identifier, not between characters. - ident.trim_matches('_').chars().all(|c| c != '_') - && ident.chars().find(|c| c.is_alphabetic()).map(|c| c.is_ascii_uppercase()).unwrap_or(true) + // Normalize the string from whatever form it's in currently, and then just make it uppercase. + let upper_snake_case = + stdx::to_lower_snake_case(ident).chars().map(|c| c.to_ascii_uppercase()).collect(); + Some(upper_snake_case) } #[cfg(test)] @@ -84,6 +138,7 @@ mod tests { check(to_lower_snake_case, "UPPER_SNAKE_CASE", expect![["upper_snake_case"]]); check(to_lower_snake_case, "Weird_Case", expect![["weird_case"]]); check(to_lower_snake_case, "CamelCase", expect![["camel_case"]]); + check(to_lower_snake_case, "lowerCamelCase", expect![["lower_camel_case"]]); } #[test] @@ -91,9 +146,19 @@ mod tests { check(to_camel_case, "CamelCase", expect![[""]]); check(to_camel_case, "CamelCase_", expect![[""]]); check(to_camel_case, "_CamelCase", expect![[""]]); + check(to_camel_case, "lowerCamelCase", expect![["LowerCamelCase"]]); check(to_camel_case, "lower_snake_case", expect![["LowerSnakeCase"]]); check(to_camel_case, "UPPER_SNAKE_CASE", expect![["UpperSnakeCase"]]); check(to_camel_case, "Weird_Case", expect![["WeirdCase"]]); check(to_camel_case, "name", expect![["Name"]]); } + + #[test] + fn test_to_upper_snake_case() { + check(to_upper_snake_case, "UPPER_SNAKE_CASE", expect![[""]]); + check(to_upper_snake_case, "lower_snake_case", expect![["LOWER_SNAKE_CASE"]]); + check(to_upper_snake_case, "Weird_Case", expect![["WEIRD_CASE"]]); + check(to_upper_snake_case, "CamelCase", expect![["CAMEL_CASE"]]); + check(to_upper_snake_case, "lowerCamelCase", expect![["LOWER_CAMEL_CASE"]]); + } } From b42562b5dee4f4ce23094c7bab6406e3b00f90ad Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 07:39:35 +0300 Subject: [PATCH 11/22] Make incorrect case diagnostic work inside of functions --- crates/hir_def/src/item_scope.rs | 6 + crates/hir_ty/src/diagnostics.rs | 4 +- crates/hir_ty/src/diagnostics/decl_check.rs | 277 +++++++++++++++++--- crates/ide/src/diagnostics.rs | 26 ++ 4 files changed, 280 insertions(+), 33 deletions(-) diff --git a/crates/hir_def/src/item_scope.rs b/crates/hir_def/src/item_scope.rs index 12c24e1ca3..a8b3fe844a 100644 --- a/crates/hir_def/src/item_scope.rs +++ b/crates/hir_def/src/item_scope.rs @@ -95,6 +95,12 @@ impl ItemScope { self.impls.iter().copied() } + pub fn values( + &self, + ) -> impl Iterator + ExactSizeIterator + '_ { + self.values.values().copied() + } + pub fn visibility_of(&self, def: ModuleDefId) -> Option { self.name_of(ItemInNs::Types(def)) .or_else(|| self.name_of(ItemInNs::Values(def))) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index bd370e3b2d..40f8c8ba2d 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -281,7 +281,7 @@ impl Diagnostic for IncorrectCase { fn message(&self) -> String { format!( - "{} `{}` should have a {} name, e.g. `{}`", + "{} `{}` should have {} name, e.g. `{}`", self.ident_type, self.ident_text, self.expected_case.to_string(), @@ -339,6 +339,8 @@ mod tests { let impl_data = self.impl_data(impl_id); for item in impl_data.items.iter() { if let AssocItemId::FunctionId(f) = item { + let mut sink = DiagnosticSinkBuilder::new().build(&mut cb); + validate_module_item(self, ModuleDefId::FunctionId(*f), &mut sink); fns.push(*f) } } diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 7fc9c564e1..d1c51849a3 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -5,23 +5,13 @@ //! - enum fields (e.g. `enum Foo { Variant { field: u8 } }`) //! - function/method arguments (e.g. `fn foo(arg: u8)`) -// TODO: Temporary, to not see warnings until module is somewhat complete. -// If you see these lines in the pull request, feel free to call me stupid :P. -#![allow(dead_code, unused_imports, unused_variables)] - mod str_helpers; -use std::sync::Arc; - use hir_def::{ adt::VariantData, - body::Body, - db::DefDatabase, - expr::{Expr, ExprId, UnaryOp}, - item_tree::ItemTreeNode, - resolver::{resolver_for_expr, ResolveValueResult, ValueNs}, + expr::{Pat, PatId}, src::HasSource, - AdtId, EnumId, FunctionId, Lookup, ModuleDefId, StructId, + AdtId, ConstId, EnumId, FunctionId, Lookup, ModuleDefId, StaticId, StructId, }; use hir_expand::{ diagnostics::DiagnosticSink, @@ -35,8 +25,6 @@ use syntax::{ use crate::{ db::HirDatabase, diagnostics::{decl_check::str_helpers::*, CaseType, IncorrectCase}, - lower::CallableDefId, - ApplicationTy, InferenceResult, Ty, TypeCtor, }; pub(super) struct DeclValidator<'a, 'b: 'a> { @@ -64,12 +52,25 @@ impl<'a, 'b> DeclValidator<'a, 'b> { match self.owner { ModuleDefId::FunctionId(func) => self.validate_func(db, func), ModuleDefId::AdtId(adt) => self.validate_adt(db, adt), + ModuleDefId::ConstId(const_id) => self.validate_const(db, const_id), + ModuleDefId::StaticId(static_id) => self.validate_static(db, static_id), _ => return, } } + fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) { + match adt { + AdtId::StructId(struct_id) => self.validate_struct(db, struct_id), + AdtId::EnumId(enum_id) => self.validate_enum(db, enum_id), + AdtId::UnionId(_) => { + // Unions aren't yet supported by this validator. + } + } + } + fn validate_func(&mut self, db: &dyn HirDatabase, func: FunctionId) { let data = db.function_data(func); + let body = db.body(func.into()); // 1. Check the function name. let function_name = data.name.to_string(); @@ -87,11 +88,18 @@ impl<'a, 'b> DeclValidator<'a, 'b> { // 2. Check the param names. let mut fn_param_replacements = Vec::new(); - for param_name in data.param_names.iter().cloned().filter_map(|i| i) { + for pat_id in body.params.iter().cloned() { + let pat = &body[pat_id]; + + let param_name = match pat { + Pat::Bind { name, .. } => name, + _ => continue, + }; + let name = param_name.to_string(); if let Some(new_name) = to_lower_snake_case(&name) { let replacement = Replacement { - current_name: param_name, + current_name: param_name.clone(), suggested_text: new_name, expected_case: CaseType::LowerSnakeCase, }; @@ -99,13 +107,45 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } } - // 3. If there is at least one element to spawn a warning on, go to the source map and generate a warning. + // 3. Check the patterns inside the function body. + let mut pats_replacements = Vec::new(); + + for (pat_idx, pat) in body.pats.iter() { + if body.params.contains(&pat_idx) { + // We aren't interested in function parameters, we've processed them above. + continue; + } + + let bind_name = match pat { + Pat::Bind { name, .. } => name, + _ => continue, + }; + + let name = bind_name.to_string(); + if let Some(new_name) = to_lower_snake_case(&name) { + let replacement = Replacement { + current_name: bind_name.clone(), + suggested_text: new_name, + expected_case: CaseType::LowerSnakeCase, + }; + pats_replacements.push((pat_idx, replacement)); + } + } + + // 4. If there is at least one element to spawn a warning on, go to the source map and generate a warning. self.create_incorrect_case_diagnostic_for_func( func, db, fn_name_replacement, fn_param_replacements, - ) + ); + self.create_incorrect_case_diagnostic_for_variables(func, db, pats_replacements); + + // 5. Recursively validate inner scope items, such as static variables and constants. + for (item_id, _) in body.item_scope.values() { + let mut validator = DeclValidator::new(item_id, self.sink); + validator.validate_item(db); + } } /// Given the information about incorrect names in the function declaration, looks up into the source code @@ -125,6 +165,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { let fn_loc = func.lookup(db.upcast()); let fn_src = fn_loc.source(db.upcast()); + // 1. Diagnostic for function name. if let Some(replacement) = fn_name_replacement { let ast_ptr = if let Some(name) = fn_src.value.name() { name @@ -150,6 +191,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { self.sink.push(diagnostic); } + // 2. Diagnostics for function params. let fn_params_list = match fn_src.value.param_list() { Some(params) => params, None => { @@ -197,12 +239,38 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } } - fn validate_adt(&mut self, db: &dyn HirDatabase, adt: AdtId) { - match adt { - AdtId::StructId(struct_id) => self.validate_struct(db, struct_id), - AdtId::EnumId(enum_id) => self.validate_enum(db, enum_id), - AdtId::UnionId(_) => { - // Unions aren't yet supported by this validator. + /// Given the information about incorrect variable names, looks up into the source code + /// for exact locations and adds diagnostics into the sink. + fn create_incorrect_case_diagnostic_for_variables( + &mut self, + func: FunctionId, + db: &dyn HirDatabase, + pats_replacements: Vec<(PatId, Replacement)>, + ) { + // XXX: only look at source_map if we do have missing fields + if pats_replacements.is_empty() { + return; + } + + let (_, source_map) = db.body_with_source_map(func.into()); + + for (id, replacement) in pats_replacements { + if let Ok(source_ptr) = source_map.pat_syntax(id) { + if let Some(expr) = source_ptr.value.as_ref().left() { + let root = source_ptr.file_syntax(db.upcast()); + if let ast::Pat::IdentPat(ident_pat) = expr.to_node(&root) { + let diagnostic = IncorrectCase { + file: source_ptr.file_id, + ident_type: "Variable".to_string(), + ident: AstPtr::new(&ident_pat).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } + } } } } @@ -246,7 +314,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { db, struct_name_replacement, struct_fields_replacements, - ) + ); } /// Given the information about incorrect names in the struct declaration, looks up into the source code @@ -464,6 +532,86 @@ impl<'a, 'b> DeclValidator<'a, 'b> { self.sink.push(diagnostic); } } + + fn validate_const(&mut self, db: &dyn HirDatabase, const_id: ConstId) { + let data = db.const_data(const_id); + + let name = match &data.name { + Some(name) => name, + None => return, + }; + + let const_name = name.to_string(); + let replacement = if let Some(new_name) = to_upper_snake_case(&const_name) { + Replacement { + current_name: name.clone(), + suggested_text: new_name, + expected_case: CaseType::UpperSnakeCase, + } + } else { + // Nothing to do here. + return; + }; + + let const_loc = const_id.lookup(db.upcast()); + let const_src = const_loc.source(db.upcast()); + + let ast_ptr = match const_src.value.name() { + Some(name) => name, + None => return, + }; + + let diagnostic = IncorrectCase { + file: const_src.file_id, + ident_type: "Constant".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } + + fn validate_static(&mut self, db: &dyn HirDatabase, static_id: StaticId) { + let data = db.static_data(static_id); + + let name = match &data.name { + Some(name) => name, + None => return, + }; + + let static_name = name.to_string(); + let replacement = if let Some(new_name) = to_upper_snake_case(&static_name) { + Replacement { + current_name: name.clone(), + suggested_text: new_name, + expected_case: CaseType::UpperSnakeCase, + } + } else { + // Nothing to do here. + return; + }; + + let static_loc = static_id.lookup(db.upcast()); + let static_src = static_loc.source(db.upcast()); + + let ast_ptr = match static_src.value.name() { + Some(name) => name, + None => return, + }; + + let diagnostic = IncorrectCase { + file: static_src.file_id, + ident_type: "Static variable".to_string(), + ident: AstPtr::new(&ast_ptr).into(), + expected_case: replacement.expected_case, + ident_text: replacement.current_name.to_string(), + suggested_text: replacement.suggested_text, + }; + + self.sink.push(diagnostic); + } } fn names_equal(left: Option, right: &Name) -> bool { @@ -491,7 +639,7 @@ mod tests { check_diagnostics( r#" fn NonSnakeCaseName() {} -// ^^^^^^^^^^^^^^^^ Function `NonSnakeCaseName` should have a snake_case name, e.g. `non_snake_case_name` +// ^^^^^^^^^^^^^^^^ Function `NonSnakeCaseName` should have snake_case name, e.g. `non_snake_case_name` "#, ); } @@ -501,10 +649,24 @@ fn NonSnakeCaseName() {} check_diagnostics( r#" fn foo(SomeParam: u8) {} - // ^^^^^^^^^ Argument `SomeParam` should have a snake_case name, e.g. `some_param` + // ^^^^^^^^^ Argument `SomeParam` should have snake_case name, e.g. `some_param` fn foo2(ok_param: &str, CAPS_PARAM: u8) {} - // ^^^^^^^^^^ Argument `CAPS_PARAM` should have a snake_case name, e.g. `caps_param` + // ^^^^^^^^^^ Argument `CAPS_PARAM` should have snake_case name, e.g. `caps_param` +"#, + ); + } + + #[test] + fn incorrect_variable_names() { + check_diagnostics( + r#" +fn foo() { + let SOME_VALUE = 10; + // ^^^^^^^^^^ Variable `SOME_VALUE` should have a snake_case name, e.g. `some_value` + let AnotherValue = 20; + // ^^^^^^^^^^^^ Variable `AnotherValue` should have snake_case name, e.g. `another_value` +} "#, ); } @@ -514,7 +676,7 @@ fn foo2(ok_param: &str, CAPS_PARAM: u8) {} check_diagnostics( r#" struct non_camel_case_name {} - // ^^^^^^^^^^^^^^^^^^^ Structure `non_camel_case_name` should have a CamelCase name, e.g. `NonCamelCaseName` + // ^^^^^^^^^^^^^^^^^^^ Structure `non_camel_case_name` should have CamelCase name, e.g. `NonCamelCaseName` "#, ); } @@ -524,7 +686,7 @@ struct non_camel_case_name {} check_diagnostics( r#" struct SomeStruct { SomeField: u8 } - // ^^^^^^^^^ Field `SomeField` should have a snake_case name, e.g. `some_field` + // ^^^^^^^^^ Field `SomeField` should have snake_case name, e.g. `some_field` "#, ); } @@ -534,7 +696,7 @@ struct SomeStruct { SomeField: u8 } check_diagnostics( r#" enum some_enum { Val(u8) } - // ^^^^^^^^^ Enum `some_enum` should have a CamelCase name, e.g. `SomeEnum` + // ^^^^^^^^^ Enum `some_enum` should have CamelCase name, e.g. `SomeEnum` "#, ); } @@ -544,7 +706,58 @@ enum some_enum { Val(u8) } check_diagnostics( r#" enum SomeEnum { SOME_VARIANT(u8) } - // ^^^^^^^^^^^^ Variant `SOME_VARIANT` should have a CamelCase name, e.g. `SomeVariant` + // ^^^^^^^^^^^^ Variant `SOME_VARIANT` should have CamelCase name, e.g. `SomeVariant` +"#, + ); + } + + #[test] + fn incorrect_const_name() { + check_diagnostics( + r#" +const some_weird_const: u8 = 10; + // ^^^^^^^^^^^^^^^^ Constant `some_weird_const` should have UPPER_SNAKE_CASE name, e.g. `SOME_WEIRD_CONST` + +fn func() { + const someConstInFunc: &str = "hi there"; + // ^^^^^^^^^^^^^^^ Constant `someConstInFunc` should have UPPER_SNAKE_CASE name, e.g. `SOME_CONST_IN_FUNC` + +} +"#, + ); + } + + #[test] + fn incorrect_static_name() { + check_diagnostics( + r#" +static some_weird_const: u8 = 10; + // ^^^^^^^^^^^^^^^^ Static variable `some_weird_const` should have UPPER_SNAKE_CASE name, e.g. `SOME_WEIRD_CONST` + +fn func() { + static someConstInFunc: &str = "hi there"; + // ^^^^^^^^^^^^^^^ Static variable `someConstInFunc` should have UPPER_SNAKE_CASE name, e.g. `SOME_CONST_IN_FUNC` +} +"#, + ); + } + + #[test] + fn fn_inside_impl_struct() { + check_diagnostics( + r#" +struct someStruct; + // ^^^^^^^^^^ Structure `someStruct` should have CamelCase name, e.g. `SomeStruct` + +impl someStruct { + fn SomeFunc(&self) { + // ^^^^^^^^ Function `SomeFunc` should have snake_case name, e.g. `some_func` + static someConstInFunc: &str = "hi there"; + // ^^^^^^^^^^^^^^^ Static variable `someConstInFunc` should have UPPER_SNAKE_CASE name, e.g. `SOME_CONST_IN_FUNC` + let WHY_VAR_IS_CAPS = 10; + // ^^^^^^^^^^^^^^^ Variable `WHY_VAR_IS_CAPS` should have snake_case name, e.g. `why_var_is_caps` + } +} "#, ); } diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index ad1b265fdf..70d5cbd383 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -877,6 +877,32 @@ pub fn SomeFn<|>(val: u8) -> u8 { pub fn some_fn(val: u8) -> u8 { if val != 0 { some_fn(val - 1) } else { val } } +"#, + ); + + check_fixes( + r#" +fn some_fn() { + let whatAWeird_Formatting<|> = 10; + another_func(whatAWeird_Formatting); +} +"#, + r#" +fn some_fn() { + let what_a_weird_formatting = 10; + another_func(what_a_weird_formatting); +} +"#, + ); + } + + #[test] + fn test_uppercase_const_no_diagnostics() { + check_no_diagnostics( + r#" +fn foo() { + const ANOTHER_ITEM<|>: &str = "some_item"; +} "#, ); } From cfbee8d3a35f81e710b17e48b8018cd6076a8133 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 07:42:06 +0300 Subject: [PATCH 12/22] Remove previously added parameter names from the function data --- crates/hir_def/src/data.rs | 2 -- crates/hir_def/src/item_tree.rs | 2 -- crates/hir_def/src/item_tree/lower.rs | 14 -------------- 3 files changed, 18 deletions(-) diff --git a/crates/hir_def/src/data.rs b/crates/hir_def/src/data.rs index 733db2eaca..ff1ef0df64 100644 --- a/crates/hir_def/src/data.rs +++ b/crates/hir_def/src/data.rs @@ -19,7 +19,6 @@ use crate::{ #[derive(Debug, Clone, PartialEq, Eq)] pub struct FunctionData { pub name: Name, - pub param_names: Vec>, pub params: Vec, pub ret_type: TypeRef, pub attrs: Attrs, @@ -40,7 +39,6 @@ impl FunctionData { Arc::new(FunctionData { name: func.name.clone(), - param_names: func.param_names.to_vec(), params: func.params.to_vec(), ret_type: func.ret_type.clone(), attrs: item_tree.attrs(ModItem::from(loc.id.value).into()).clone(), diff --git a/crates/hir_def/src/item_tree.rs b/crates/hir_def/src/item_tree.rs index ca502ce2ba..8a1121bbdf 100644 --- a/crates/hir_def/src/item_tree.rs +++ b/crates/hir_def/src/item_tree.rs @@ -507,8 +507,6 @@ pub struct Function { pub has_self_param: bool, pub has_body: bool, pub is_unsafe: bool, - /// List of function parameters names. Does not include `self`. - pub param_names: Box<[Option]>, pub params: Box<[TypeRef]>, pub is_varargs: bool, pub ret_type: TypeRef, diff --git a/crates/hir_def/src/item_tree/lower.rs b/crates/hir_def/src/item_tree/lower.rs index 2ffa46ac0e..3328639cfe 100644 --- a/crates/hir_def/src/item_tree/lower.rs +++ b/crates/hir_def/src/item_tree/lower.rs @@ -283,7 +283,6 @@ impl Ctx { let name = func.name()?.as_name(); let mut params = Vec::new(); - let mut param_names = Vec::new(); let mut has_self_param = false; if let Some(param_list) = func.param_list() { if let Some(self_param) = param_list.self_param() { @@ -306,18 +305,6 @@ impl Ctx { has_self_param = true; } for param in param_list.params() { - let param_name = param - .pat() - .map(|name| { - if let ast::Pat::IdentPat(ident) = name { - Some(ident.name()?.as_name()) - } else { - None - } - }) - .flatten(); - param_names.push(param_name); - let type_ref = TypeRef::from_ast_opt(&self.body_ctx, param.ty()); params.push(type_ref); } @@ -354,7 +341,6 @@ impl Ctx { has_body, is_unsafe: func.unsafe_token().is_some(), params: params.into_boxed_slice(), - param_names: param_names.into_boxed_slice(), is_varargs, ret_type, ast_id, From 45ac2b2edec05e417124ebfc2e61ec2a5117f4d5 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 08:53:34 +0300 Subject: [PATCH 13/22] Code style adjustments --- crates/hir_ty/src/diagnostics/decl_check.rs | 52 ++++++++++++++++++- .../src/diagnostics/decl_check/str_helpers.rs | 38 ++++++++++++-- crates/hir_ty/src/diagnostics/unsafe_check.rs | 6 +-- 3 files changed, 88 insertions(+), 8 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index d1c51849a3..3a95f1b82a 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -19,7 +19,7 @@ use hir_expand::{ }; use syntax::{ ast::{self, NameOwner}, - AstPtr, + AstNode, AstPtr, }; use crate::{ @@ -259,6 +259,21 @@ impl<'a, 'b> DeclValidator<'a, 'b> { if let Some(expr) = source_ptr.value.as_ref().left() { let root = source_ptr.file_syntax(db.upcast()); if let ast::Pat::IdentPat(ident_pat) = expr.to_node(&root) { + let parent = match ident_pat.syntax().parent() { + Some(parent) => parent, + None => continue, + }; + + // We have to check that it's either `let var = ...` or `Variant(_) @ var` statement, + // because e.g. match arms are patterns as well. + // In other words, we check that it's a named variable binding. + if !ast::LetStmt::cast(parent.clone()).is_some() + && !ast::IdentPat::cast(parent).is_some() + { + // This pattern is not an actual variable declaration, e.g. `Some(val) => {..}` match arm. + continue; + } + let diagnostic = IncorrectCase { file: source_ptr.file_id, ident_type: "Variable".to_string(), @@ -663,7 +678,7 @@ fn foo2(ok_param: &str, CAPS_PARAM: u8) {} r#" fn foo() { let SOME_VALUE = 10; - // ^^^^^^^^^^ Variable `SOME_VALUE` should have a snake_case name, e.g. `some_value` + // ^^^^^^^^^^ Variable `SOME_VALUE` should have snake_case name, e.g. `some_value` let AnotherValue = 20; // ^^^^^^^^^^^^ Variable `AnotherValue` should have snake_case name, e.g. `another_value` } @@ -758,6 +773,39 @@ impl someStruct { // ^^^^^^^^^^^^^^^ Variable `WHY_VAR_IS_CAPS` should have snake_case name, e.g. `why_var_is_caps` } } +"#, + ); + } + + #[test] + fn no_diagnostic_for_enum_varinats() { + check_diagnostics( + r#" +enum Option { Some, None } + +fn main() { + match Option::None { + None => (), + Some => (), + } +} +"#, + ); + } + + #[test] + fn non_let_bind() { + check_diagnostics( + r#" +enum Option { Some, None } + +fn main() { + match Option::None { + None @ SOME_VAR => (), + // ^^^^^^^^ Variable `SOME_VAR` should have snake_case name, e.g. `some_var` + Some => (), + } +} "#, ); } diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs index e3826909bb..8f70c5e846 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -1,3 +1,6 @@ +//! Functions for string case manipulation, such as detecting the identifier case, +//! and converting it into appropriate form. + #[derive(Debug)] enum DetectedCase { LowerCamelCase, @@ -44,6 +47,8 @@ fn detect_case(ident: &str) -> DetectedCase { } } +/// Converts an identifier to an UpperCamelCase form. +/// Returns `None` if the string is already is UpperCamelCase. pub fn to_camel_case(ident: &str) -> Option { let detected_case = detect_case(ident); @@ -87,9 +92,17 @@ pub fn to_camel_case(ident: &str) -> Option { } } - Some(output) + if output == 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 UPPER_SNAKE_CASE. + None + } else { + Some(output) + } } +/// Converts an identifier to a lower_snake_case form. +/// Returns `None` if the string is already is lower_snake_case. pub fn to_lower_snake_case(ident: &str) -> Option { // First, assume that it's UPPER_SNAKE_CASE. match detect_case(ident) { @@ -102,9 +115,18 @@ pub fn to_lower_snake_case(ident: &str) -> Option { // Otherwise, assume that it's CamelCase. let lower_snake_case = stdx::to_lower_snake_case(ident); - Some(lower_snake_case) + + 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. +/// Returns `None` if the string is already is UPPER_SNAKE_CASE. pub fn to_upper_snake_case(ident: &str) -> Option { match detect_case(ident) { DetectedCase::UpperSnakeCase => return None, @@ -117,7 +139,14 @@ pub fn to_upper_snake_case(ident: &str) -> Option { // Normalize the string from whatever form it's in currently, and then just make it uppercase. let upper_snake_case = stdx::to_lower_snake_case(ident).chars().map(|c| c.to_ascii_uppercase()).collect(); - Some(upper_snake_case) + + if upper_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 UPPER_SNAKE_CASE. + None + } else { + Some(upper_snake_case) + } } #[cfg(test)] @@ -139,6 +168,7 @@ mod tests { check(to_lower_snake_case, "Weird_Case", expect![["weird_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, "a", expect![[""]]); } #[test] @@ -151,6 +181,7 @@ mod tests { check(to_camel_case, "UPPER_SNAKE_CASE", expect![["UpperSnakeCase"]]); check(to_camel_case, "Weird_Case", expect![["WeirdCase"]]); check(to_camel_case, "name", expect![["Name"]]); + check(to_camel_case, "A", expect![[""]]); } #[test] @@ -160,5 +191,6 @@ mod tests { check(to_upper_snake_case, "Weird_Case", expect![["WEIRD_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, "A", expect![[""]]); } } diff --git a/crates/hir_ty/src/diagnostics/unsafe_check.rs b/crates/hir_ty/src/diagnostics/unsafe_check.rs index 61ffbf5d15..21a121aad7 100644 --- a/crates/hir_ty/src/diagnostics/unsafe_check.rs +++ b/crates/hir_ty/src/diagnostics/unsafe_check.rs @@ -190,13 +190,13 @@ struct Ty { a: u8, } -static mut static_mut: Ty = Ty { a: 0 }; +static mut STATIC_MUT: Ty = Ty { a: 0 }; fn main() { - let x = static_mut.a; + let x = STATIC_MUT.a; //^^^^^^^^^^ This operation is unsafe and requires an unsafe function or block unsafe { - let x = static_mut.a; + let x = STATIC_MUT.a; } } "#, From 2a72f876d655da086e436838fdbc797a2ef71ece Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 09:04:28 +0300 Subject: [PATCH 14/22] Fix issues with match arm bindings --- crates/hir_ty/src/diagnostics/decl_check.rs | 19 ++++++++++++------- 1 file changed, 12 insertions(+), 7 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 3a95f1b82a..28ce15773b 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -263,13 +263,18 @@ impl<'a, 'b> DeclValidator<'a, 'b> { Some(parent) => parent, None => continue, }; + let name_ast = match ident_pat.name() { + Some(name_ast) => name_ast, + None => continue, + }; - // We have to check that it's either `let var = ...` or `Variant(_) @ var` statement, + // We have to check that it's either `let var = ...` or `var @ Variant(_)` statement, // because e.g. match arms are patterns as well. // In other words, we check that it's a named variable binding. - if !ast::LetStmt::cast(parent.clone()).is_some() - && !ast::IdentPat::cast(parent).is_some() - { + let is_binding = ast::LetStmt::cast(parent.clone()).is_some() + || (ast::MatchArm::cast(parent).is_some() + && ident_pat.at_token().is_some()); + if !is_binding { // This pattern is not an actual variable declaration, e.g. `Some(val) => {..}` match arm. continue; } @@ -277,7 +282,7 @@ impl<'a, 'b> DeclValidator<'a, 'b> { let diagnostic = IncorrectCase { file: source_ptr.file_id, ident_type: "Variable".to_string(), - ident: AstPtr::new(&ident_pat).into(), + ident: AstPtr::new(&name_ast).into(), expected_case: replacement.expected_case, ident_text: replacement.current_name.to_string(), suggested_text: replacement.suggested_text, @@ -801,8 +806,8 @@ enum Option { Some, None } fn main() { match Option::None { - None @ SOME_VAR => (), - // ^^^^^^^^ Variable `SOME_VAR` should have snake_case name, e.g. `some_var` + SOME_VAR @ None => (), + // ^^^^^^^^ Variable `SOME_VAR` should have snake_case name, e.g. `some_var` Some => (), } } From 50a147dcdfd0df462f0c24e5d7bcfe60abadac32 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 09:12:00 +0300 Subject: [PATCH 15/22] Apply case check diagnostic to impl items --- crates/hir/src/code_model.rs | 3 ++- crates/ide/src/diagnostics.rs | 24 ++++++++++++++++++++++++ 2 files changed, 26 insertions(+), 1 deletion(-) diff --git a/crates/hir/src/code_model.rs b/crates/hir/src/code_model.rs index c134356ef1..fb85041fdb 100644 --- a/crates/hir/src/code_model.rs +++ b/crates/hir/src/code_model.rs @@ -781,7 +781,8 @@ impl Function { } pub fn diagnostics(self, db: &dyn HirDatabase, sink: &mut DiagnosticSink) { - hir_ty::diagnostics::validate_body(db, self.id.into(), sink) + hir_ty::diagnostics::validate_module_item(db, self.id.into(), sink); + hir_ty::diagnostics::validate_body(db, self.id.into(), sink); } /// Whether this function declaration has a definition. diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index 70d5cbd383..8e508639be 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -903,6 +903,30 @@ fn some_fn() { fn foo() { const ANOTHER_ITEM<|>: &str = "some_item"; } +"#, + ); + } + + #[test] + fn test_rename_incorrect_case_struct_method() { + check_fixes( + r#" +pub struct TestStruct; + +impl TestStruct { + pub fn SomeFn<|>() -> TestStruct { + TestStruct + } +} +"#, + r#" +pub struct TestStruct; + +impl TestStruct { + pub fn some_fn() -> TestStruct { + TestStruct + } +} "#, ); } From 9ea57cac0e9779ac0749ef568eeb3977fe3adacd Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Sun, 4 Oct 2020 09:26:39 +0300 Subject: [PATCH 16/22] Fix code style issues --- crates/hir_ty/src/diagnostics.rs | 2 +- crates/hir_ty/src/diagnostics/decl_check.rs | 8 ++++++-- crates/ide/src/diagnostics.rs | 1 - 3 files changed, 7 insertions(+), 4 deletions(-) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index 40f8c8ba2d..f2e06495e3 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -21,7 +21,7 @@ pub fn validate_module_item( owner: ModuleDefId, sink: &mut DiagnosticSink<'_>, ) { - let _p = profile::span("validate_body"); + let _p = profile::span("validate_module_item"); let mut validator = decl_check::DeclValidator::new(owner, sink); validator.validate_item(db); } diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 28ce15773b..4c20921e5d 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -1,9 +1,14 @@ //! Provides validators for the item declarations. +//! //! This includes the following items: +//! //! - variable bindings (e.g. `let x = foo();`) //! - struct fields (e.g. `struct Foo { field: u8 }`) -//! - enum fields (e.g. `enum Foo { Variant { field: u8 } }`) +//! - enum variants (e.g. `enum Foo { Variant { field: u8 } }`) //! - function/method arguments (e.g. `fn foo(arg: u8)`) +//! - constants (e.g. `const FOO: u8 = 10;`) +//! - static items (e.g. `static FOO: u8 = 10;`) +//! - match arm bindings (e.g. `foo @ Some(_)`) mod str_helpers; @@ -48,7 +53,6 @@ impl<'a, 'b> DeclValidator<'a, 'b> { } pub(super) fn validate_item(&mut self, db: &dyn HirDatabase) { - // let def = self.owner.into(); match self.owner { ModuleDefId::FunctionId(func) => self.validate_func(db, func), ModuleDefId::AdtId(adt) => self.validate_adt(db, adt), diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index 8e508639be..add102ff26 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -135,7 +135,6 @@ fn diagnostic_with_fix(d: &D, sema: &Semantics(d: &D, sema: &Semantics) -> Diagnostic { Diagnostic { - // name: Some(d.name().into()), range: sema.diagnostics_display_range(d).range, message: d.message(), severity: Severity::WeakWarning, From f2c91fc5a8c22b8ac80f100b2b666f2dc9baa67c Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Mon, 5 Oct 2020 19:34:23 +0300 Subject: [PATCH 17/22] Apply suggestions from code review Co-authored-by: Lukas Wirth --- crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs | 4 ++-- crates/stdx/src/lib.rs | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs index 8f70c5e846..c1ab1a6759 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -13,7 +13,7 @@ enum DetectedCase { fn detect_case(ident: &str) -> DetectedCase { let trimmed_ident = ident.trim_matches('_'); let first_lowercase = - trimmed_ident.chars().next().map(|chr| chr.is_ascii_lowercase()).unwrap_or(false); + trimmed_ident.starts_with(|chr| chr.is_ascii_lowercase()); let mut has_lowercase = first_lowercase; let mut has_uppercase = false; let mut has_underscore = false; @@ -102,7 +102,7 @@ pub fn to_camel_case(ident: &str) -> Option { } /// Converts an identifier to a lower_snake_case form. -/// Returns `None` if the string is already is lower_snake_case. +/// Returns `None` if the string is already in lower_snake_case. pub fn to_lower_snake_case(ident: &str) -> Option { // First, assume that it's UPPER_SNAKE_CASE. match detect_case(ident) { diff --git a/crates/stdx/src/lib.rs b/crates/stdx/src/lib.rs index 522a9c1abd..b55de813ec 100644 --- a/crates/stdx/src/lib.rs +++ b/crates/stdx/src/lib.rs @@ -35,7 +35,7 @@ pub fn to_lower_snake_case(s: &str) -> String { // `&& prev` is required to not insert `_` before the first symbol. if c.is_ascii_uppercase() && prev { // This check is required to not translate `Weird_Case` into `weird__case`. - if buf.chars().last() != Some('_') { + if !buf.ends_with('_') { buf.push('_') } } From ebd30033b3743fafe0a0182b5ae34ffb27fe43ff Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Mon, 5 Oct 2020 20:35:52 +0300 Subject: [PATCH 18/22] Fix compilation error --- crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs index c1ab1a6759..2e1468c4cd 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs @@ -12,8 +12,7 @@ enum DetectedCase { fn detect_case(ident: &str) -> DetectedCase { let trimmed_ident = ident.trim_matches('_'); - let first_lowercase = - trimmed_ident.starts_with(|chr| chr.is_ascii_lowercase()); + 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; From 559cc970732d80e3ec624c20da4f8aac219d6b2e Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Thu, 8 Oct 2020 08:33:35 +0300 Subject: [PATCH 19/22] Add to_upper_snake_case function to stdx --- crates/hir_ty/src/diagnostics/decl_check.rs | 4 ++-- .../decl_check/{str_helpers.rs => case_conv.rs} | 3 +-- crates/stdx/src/lib.rs | 12 ++++++++++-- 3 files changed, 13 insertions(+), 6 deletions(-) rename crates/hir_ty/src/diagnostics/decl_check/{str_helpers.rs => case_conv.rs} (98%) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 4c20921e5d..901ccc94f6 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -10,7 +10,7 @@ //! - static items (e.g. `static FOO: u8 = 10;`) //! - match arm bindings (e.g. `foo @ Some(_)`) -mod str_helpers; +mod case_conv; use hir_def::{ adt::VariantData, @@ -29,7 +29,7 @@ use syntax::{ use crate::{ db::HirDatabase, - diagnostics::{decl_check::str_helpers::*, CaseType, IncorrectCase}, + diagnostics::{decl_check::case_conv::*, CaseType, IncorrectCase}, }; pub(super) struct DeclValidator<'a, 'b: 'a> { diff --git a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs b/crates/hir_ty/src/diagnostics/decl_check/case_conv.rs similarity index 98% rename from crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs rename to crates/hir_ty/src/diagnostics/decl_check/case_conv.rs index 2e1468c4cd..3800f2a6b3 100644 --- a/crates/hir_ty/src/diagnostics/decl_check/str_helpers.rs +++ b/crates/hir_ty/src/diagnostics/decl_check/case_conv.rs @@ -136,8 +136,7 @@ pub fn to_upper_snake_case(ident: &str) -> Option { } // Normalize the string from whatever form it's in currently, and then just make it uppercase. - let upper_snake_case = - stdx::to_lower_snake_case(ident).chars().map(|c| c.to_ascii_uppercase()).collect(); + let upper_snake_case = stdx::to_upper_snake_case(ident); if upper_snake_case == ident { // While we didn't detect the correct case at the beginning, there diff --git a/crates/stdx/src/lib.rs b/crates/stdx/src/lib.rs index b55de813ec..59d89f47d1 100644 --- a/crates/stdx/src/lib.rs +++ b/crates/stdx/src/lib.rs @@ -28,7 +28,7 @@ pub fn timeit(label: &'static str) -> impl Drop { Guard { label, start: Instant::now() } } -pub fn to_lower_snake_case(s: &str) -> String { +fn to_snake_case char>(s: &str, change_case: F) -> String { let mut buf = String::with_capacity(s.len()); let mut prev = false; for c in s.chars() { @@ -41,11 +41,19 @@ pub fn to_lower_snake_case(s: &str) -> String { } prev = true; - buf.push(c.to_ascii_lowercase()); + buf.push(change_case(&c)); } buf } +pub fn to_lower_snake_case(s: &str) -> String { + to_snake_case(s, char::to_ascii_lowercase) +} + +pub fn to_upper_snake_case(s: &str) -> String { + to_snake_case(s, char::to_ascii_uppercase) +} + pub fn replace(buf: &mut String, from: char, to: &str) { if !buf.contains(from) { return; From 66cea8cbaa3320653e760e7b4ce839e055976acf Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Thu, 8 Oct 2020 08:40:30 +0300 Subject: [PATCH 20/22] Replace 'if let' with 'match' in decl_check.rs --- crates/hir_ty/src/diagnostics/decl_check.rs | 63 +++++++++++---------- 1 file changed, 33 insertions(+), 30 deletions(-) diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 901ccc94f6..1f9386b751 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -171,16 +171,17 @@ impl<'a, 'b> DeclValidator<'a, 'b> { // 1. Diagnostic for function name. if let Some(replacement) = fn_name_replacement { - let ast_ptr = if let Some(name) = fn_src.value.name() { - name - } else { - // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. - log::error!( - "Replacement ({:?}) was generated for a function without a name: {:?}", - replacement, - fn_src - ); - return; + let ast_ptr = match fn_src.value.name() { + Some(name) => name, + None => { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement ({:?}) was generated for a function without a name: {:?}", + replacement, + fn_src + ); + return; + } }; let diagnostic = IncorrectCase { @@ -359,16 +360,17 @@ impl<'a, 'b> DeclValidator<'a, 'b> { let struct_src = struct_loc.source(db.upcast()); if let Some(replacement) = struct_name_replacement { - let ast_ptr = if let Some(name) = struct_src.value.name() { - name - } else { - // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. - log::error!( - "Replacement ({:?}) was generated for a structure without a name: {:?}", - replacement, - struct_src - ); - return; + let ast_ptr = match struct_src.value.name() { + Some(name) => name, + None => { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement ({:?}) was generated for a structure without a name: {:?}", + replacement, + struct_src + ); + return; + } }; let diagnostic = IncorrectCase { @@ -486,16 +488,17 @@ impl<'a, 'b> DeclValidator<'a, 'b> { let enum_src = enum_loc.source(db.upcast()); if let Some(replacement) = enum_name_replacement { - let ast_ptr = if let Some(name) = enum_src.value.name() { - name - } else { - // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. - log::error!( - "Replacement ({:?}) was generated for a enum without a name: {:?}", - replacement, - enum_src - ); - return; + let ast_ptr = match enum_src.value.name() { + Some(name) => name, + None => { + // We don't want rust-analyzer to panic over this, but it is definitely some kind of error in the logic. + log::error!( + "Replacement ({:?}) was generated for a enum without a name: {:?}", + replacement, + enum_src + ); + return; + } }; let diagnostic = IncorrectCase { From fb0ab9f7456018ff0bac628e05366f976c5af1a7 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Thu, 8 Oct 2020 09:27:38 +0300 Subject: [PATCH 21/22] Keep SyntaxNodePtr::range private --- crates/hir_ty/src/diagnostics.rs | 4 ++-- crates/hir_ty/src/diagnostics/decl_check.rs | 13 +++++++++++-- crates/ide/src/diagnostics/fixes.rs | 5 ++++- crates/syntax/src/ptr.rs | 4 ---- 4 files changed, 17 insertions(+), 9 deletions(-) diff --git a/crates/hir_ty/src/diagnostics.rs b/crates/hir_ty/src/diagnostics.rs index f2e06495e3..dfe98571e7 100644 --- a/crates/hir_ty/src/diagnostics.rs +++ b/crates/hir_ty/src/diagnostics.rs @@ -267,7 +267,7 @@ impl fmt::Display for CaseType { #[derive(Debug)] pub struct IncorrectCase { pub file: HirFileId, - pub ident: SyntaxNodePtr, + pub ident: AstPtr, pub expected_case: CaseType, pub ident_type: String, pub ident_text: String, @@ -290,7 +290,7 @@ impl Diagnostic for IncorrectCase { } fn display_source(&self) -> InFile { - InFile::new(self.file, self.ident.clone()) + InFile::new(self.file, self.ident.clone().into()) } fn as_any(&self) -> &(dyn Any + Send + 'static) { diff --git a/crates/hir_ty/src/diagnostics/decl_check.rs b/crates/hir_ty/src/diagnostics/decl_check.rs index 1f9386b751..f987636fe5 100644 --- a/crates/hir_ty/src/diagnostics/decl_check.rs +++ b/crates/hir_ty/src/diagnostics/decl_check.rs @@ -213,12 +213,21 @@ impl<'a, 'b> DeclValidator<'a, 'b> { for param_to_rename in fn_param_replacements { // We assume that parameters in replacement are in the same order as in the // actual params list, but just some of them (ones that named correctly) are skipped. - let ast_ptr = loop { + let ast_ptr: ast::Name = loop { match fn_params_iter.next() { Some(element) if pat_equals_to_name(element.pat(), ¶m_to_rename.current_name) => { - break element.pat().unwrap() + if let ast::Pat::IdentPat(pat) = element.pat().unwrap() { + break pat.name().unwrap(); + } else { + // This is critical. If we consider this parameter the expected one, + // it **must** have a name. + panic!( + "Pattern {:?} equals to expected replacement {:?}, but has no name", + element, param_to_rename + ); + } } Some(_) => {} None => { diff --git a/crates/ide/src/diagnostics/fixes.rs b/crates/ide/src/diagnostics/fixes.rs index 286ef07850..b47fe04691 100644 --- a/crates/ide/src/diagnostics/fixes.rs +++ b/crates/ide/src/diagnostics/fixes.rs @@ -104,8 +104,11 @@ impl DiagnosticWithFix for MissingOkInTailExpr { impl DiagnosticWithFix for IncorrectCase { fn fix(&self, sema: &Semantics) -> Option { + let root = sema.db.parse_or_expand(self.file)?; + let name_node = self.ident.to_node(&root); + let file_id = self.file.original_file(sema.db); - let offset = self.ident.text_range().start(); + let offset = name_node.syntax().text_range().start(); let file_position = FilePosition { file_id, offset }; let rename_changes = rename_with_semantics(sema, file_position, &self.suggested_text)?; diff --git a/crates/syntax/src/ptr.rs b/crates/syntax/src/ptr.rs index 34e20464a7..d3fb7a5d98 100644 --- a/crates/syntax/src/ptr.rs +++ b/crates/syntax/src/ptr.rs @@ -23,10 +23,6 @@ impl SyntaxNodePtr { SyntaxNodePtr { range: node.text_range(), kind: node.kind() } } - pub fn text_range(&self) -> TextRange { - self.range.clone() - } - pub fn to_node(&self, root: &SyntaxNode) -> SyntaxNode { assert!(root.parent().is_none()); successors(Some(root.clone()), |node| { From 991d0190968662f23220d8aefaf28bd03b1dbe41 Mon Sep 17 00:00:00 2001 From: Igor Aleksanov Date: Thu, 8 Oct 2020 09:30:01 +0300 Subject: [PATCH 22/22] Use TextRange::contains_inclusive in fixes check --- crates/ide/src/diagnostics.rs | 6 ++---- crates/ide/src/diagnostics/fixes.rs | 3 ++- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/crates/ide/src/diagnostics.rs b/crates/ide/src/diagnostics.rs index add102ff26..b30cdb6edd 100644 --- a/crates/ide/src/diagnostics.rs +++ b/crates/ide/src/diagnostics.rs @@ -257,8 +257,7 @@ mod tests { assert_eq_text!(&after, &actual); assert!( - fix.fix_trigger_range.start() <= file_position.offset - && fix.fix_trigger_range.end() >= file_position.offset, + fix.fix_trigger_range.contains_inclusive(file_position.offset), "diagnostic fix range {:?} does not touch cursor position {:?}", fix.fix_trigger_range, file_position.offset @@ -288,8 +287,7 @@ mod tests { assert_eq_text!(&after, &actual); assert!( - fix.fix_trigger_range.start() <= file_position.offset - && fix.fix_trigger_range.end() >= file_position.offset, + fix.fix_trigger_range.contains_inclusive(file_position.offset), "diagnostic fix range {:?} does not touch cursor position {:?}", fix.fix_trigger_range, file_position.offset diff --git a/crates/ide/src/diagnostics/fixes.rs b/crates/ide/src/diagnostics/fixes.rs index b47fe04691..0c75e50b01 100644 --- a/crates/ide/src/diagnostics/fixes.rs +++ b/crates/ide/src/diagnostics/fixes.rs @@ -111,7 +111,8 @@ impl DiagnosticWithFix for IncorrectCase { let offset = name_node.syntax().text_range().start(); let file_position = FilePosition { file_id, offset }; - let rename_changes = rename_with_semantics(sema, file_position, &self.suggested_text)?; + let rename_changes = + rename_with_semantics(sema, file_position, &self.suggested_text).ok()?; let label = format!("Rename to {}", self.suggested_text); Some(Fix::new(&label, rename_changes.info, rename_changes.range))