mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-13 21:54:42 +00:00
Wrap inner tail expressions in MissingOkOrSomeInTailExpr
This commit is contained in:
parent
eb513c22f7
commit
1edbaa29f9
8 changed files with 69 additions and 11 deletions
|
@ -9,6 +9,8 @@ use hir_def::path::ModPath;
|
||||||
use hir_expand::{name::Name, HirFileId, InFile};
|
use hir_expand::{name::Name, HirFileId, InFile};
|
||||||
use syntax::{ast, AstPtr, SyntaxNodePtr, TextRange};
|
use syntax::{ast, AstPtr, SyntaxNodePtr, TextRange};
|
||||||
|
|
||||||
|
use crate::Type;
|
||||||
|
|
||||||
macro_rules! diagnostics {
|
macro_rules! diagnostics {
|
||||||
($($diag:ident,)*) => {
|
($($diag:ident,)*) => {
|
||||||
pub enum AnyDiagnostic {$(
|
pub enum AnyDiagnostic {$(
|
||||||
|
@ -142,6 +144,7 @@ pub struct MissingOkOrSomeInTailExpr {
|
||||||
pub expr: InFile<AstPtr<ast::Expr>>,
|
pub expr: InFile<AstPtr<ast::Expr>>,
|
||||||
// `Some` or `Ok` depending on whether the return type is Result or Option
|
// `Some` or `Ok` depending on whether the return type is Result or Option
|
||||||
pub required: String,
|
pub required: String,
|
||||||
|
pub expected: Type,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
|
|
@ -1216,7 +1216,14 @@ impl Function {
|
||||||
}
|
}
|
||||||
BodyValidationDiagnostic::MissingOkOrSomeInTailExpr { expr, required } => {
|
BodyValidationDiagnostic::MissingOkOrSomeInTailExpr { expr, required } => {
|
||||||
match source_map.expr_syntax(expr) {
|
match source_map.expr_syntax(expr) {
|
||||||
Ok(expr) => acc.push(MissingOkOrSomeInTailExpr { expr, required }.into()),
|
Ok(expr) => acc.push(
|
||||||
|
MissingOkOrSomeInTailExpr {
|
||||||
|
expr,
|
||||||
|
required,
|
||||||
|
expected: self.ret_type(db),
|
||||||
|
}
|
||||||
|
.into(),
|
||||||
|
),
|
||||||
Err(SyntheticSyntax) => (),
|
Err(SyntheticSyntax) => (),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@
|
||||||
|
|
||||||
mod deconstruct_pat;
|
mod deconstruct_pat;
|
||||||
mod pat_util;
|
mod pat_util;
|
||||||
|
|
||||||
pub(crate) mod usefulness;
|
pub(crate) mod usefulness;
|
||||||
|
|
||||||
use hir_def::{body::Body, EnumVariantId, LocalFieldId, VariantId};
|
use hir_def::{body::Body, EnumVariantId, LocalFieldId, VariantId};
|
||||||
|
|
|
@ -533,6 +533,9 @@ fn foo() ->$0 u32 {
|
||||||
5
|
5
|
||||||
// ^
|
// ^
|
||||||
}
|
}
|
||||||
|
} else if false {
|
||||||
|
0
|
||||||
|
// ^
|
||||||
} else {
|
} else {
|
||||||
match 5 {
|
match 5 {
|
||||||
6 => 100,
|
6 => 100,
|
||||||
|
|
|
@ -624,11 +624,7 @@ impl<'a> CompletionContext<'a> {
|
||||||
fn classify_name(&mut self, name: ast::Name) {
|
fn classify_name(&mut self, name: ast::Name) {
|
||||||
if let Some(bind_pat) = name.syntax().parent().and_then(ast::IdentPat::cast) {
|
if let Some(bind_pat) = name.syntax().parent().and_then(ast::IdentPat::cast) {
|
||||||
self.is_pat_or_const = Some(PatternRefutability::Refutable);
|
self.is_pat_or_const = Some(PatternRefutability::Refutable);
|
||||||
// if any of these is here our bind pat can't be a const pat anymore
|
if !bind_pat.is_simple_ident() {
|
||||||
let complex_ident_pat = bind_pat.at_token().is_some()
|
|
||||||
|| bind_pat.ref_token().is_some()
|
|
||||||
|| bind_pat.mut_token().is_some();
|
|
||||||
if complex_ident_pat {
|
|
||||||
self.is_pat_or_const = None;
|
self.is_pat_or_const = None;
|
||||||
} else {
|
} else {
|
||||||
let irrefutable_pat = bind_pat.syntax().ancestors().find_map(|node| {
|
let irrefutable_pat = bind_pat.syntax().ancestors().find_map(|node| {
|
||||||
|
|
|
@ -271,7 +271,20 @@ pub fn for_each_tail_expr(expr: &ast::Expr, cb: &mut dyn FnMut(&ast::Expr)) {
|
||||||
ast::Effect::Async(_) | ast::Effect::Try(_) | ast::Effect::Const(_) => cb(expr),
|
ast::Effect::Async(_) | ast::Effect::Try(_) | ast::Effect::Const(_) => cb(expr),
|
||||||
},
|
},
|
||||||
ast::Expr::IfExpr(if_) => {
|
ast::Expr::IfExpr(if_) => {
|
||||||
if_.blocks().for_each(|block| for_each_tail_expr(&ast::Expr::BlockExpr(block), cb))
|
let mut if_ = if_.clone();
|
||||||
|
loop {
|
||||||
|
if let Some(block) = if_.then_branch() {
|
||||||
|
for_each_tail_expr(&ast::Expr::BlockExpr(block), cb);
|
||||||
|
}
|
||||||
|
match if_.else_branch() {
|
||||||
|
Some(ast::ElseBranch::IfExpr(it)) => if_ = it,
|
||||||
|
Some(ast::ElseBranch::Block(block)) => {
|
||||||
|
for_each_tail_expr(&ast::Expr::BlockExpr(block), cb);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
None => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
ast::Expr::LoopExpr(l) => {
|
ast::Expr::LoopExpr(l) => {
|
||||||
for_each_break_expr(l.label(), l.loop_body(), &mut |b| cb(&ast::Expr::BreakExpr(b)))
|
for_each_break_expr(l.label(), l.loop_body(), &mut |b| cb(&ast::Expr::BreakExpr(b)))
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
use hir::db::AstDatabase;
|
use hir::db::AstDatabase;
|
||||||
use ide_db::{assists::Assist, source_change::SourceChange};
|
use ide_db::{assists::Assist, helpers::for_each_tail_expr, source_change::SourceChange};
|
||||||
use syntax::AstNode;
|
use syntax::AstNode;
|
||||||
use text_edit::TextEdit;
|
use text_edit::TextEdit;
|
||||||
|
|
||||||
|
@ -33,10 +33,15 @@ fn fixes(ctx: &DiagnosticsContext<'_>, d: &hir::MissingOkOrSomeInTailExpr) -> Op
|
||||||
let root = ctx.sema.db.parse_or_expand(d.expr.file_id)?;
|
let root = ctx.sema.db.parse_or_expand(d.expr.file_id)?;
|
||||||
let tail_expr = d.expr.value.to_node(&root);
|
let tail_expr = d.expr.value.to_node(&root);
|
||||||
let tail_expr_range = tail_expr.syntax().text_range();
|
let tail_expr_range = tail_expr.syntax().text_range();
|
||||||
let replacement = format!("{}({})", d.required, tail_expr.syntax());
|
let mut builder = TextEdit::builder();
|
||||||
let edit = TextEdit::replace(tail_expr_range, replacement);
|
for_each_tail_expr(&tail_expr, &mut |expr| {
|
||||||
|
if ctx.sema.type_of_expr(expr).as_ref() != Some(&d.expected) {
|
||||||
|
builder.insert(expr.syntax().text_range().start(), format!("{}(", d.required));
|
||||||
|
builder.insert(expr.syntax().text_range().end(), ")".to_string());
|
||||||
|
}
|
||||||
|
});
|
||||||
let source_change =
|
let source_change =
|
||||||
SourceChange::from_text_edit(d.expr.file_id.original_file(ctx.sema.db), edit);
|
SourceChange::from_text_edit(d.expr.file_id.original_file(ctx.sema.db), builder.finish());
|
||||||
let name = if d.required == "Ok" { "Wrap with Ok" } else { "Wrap with Some" };
|
let name = if d.required == "Ok" { "Wrap with Ok" } else { "Wrap with Some" };
|
||||||
Some(vec![fix("wrap_tail_expr", name, source_change, tail_expr_range)])
|
Some(vec![fix("wrap_tail_expr", name, source_change, tail_expr_range)])
|
||||||
}
|
}
|
||||||
|
@ -68,6 +73,35 @@ fn div(x: i32, y: i32) -> Option<i32> {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_wrap_return_type_option_tails() {
|
||||||
|
check_fix(
|
||||||
|
r#"
|
||||||
|
//- minicore: option, result
|
||||||
|
fn div(x: i32, y: i32) -> Option<i32> {
|
||||||
|
if y == 0 {
|
||||||
|
0
|
||||||
|
} else if true {
|
||||||
|
100
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}$0
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
r#"
|
||||||
|
fn div(x: i32, y: i32) -> Option<i32> {
|
||||||
|
if y == 0 {
|
||||||
|
Some(0)
|
||||||
|
} else if true {
|
||||||
|
Some(100)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"#,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_wrap_return_type() {
|
fn test_wrap_return_type() {
|
||||||
check_fix(
|
check_fix(
|
||||||
|
|
|
@ -164,6 +164,7 @@ impl ast::IfExpr {
|
||||||
pub fn then_branch(&self) -> Option<ast::BlockExpr> {
|
pub fn then_branch(&self) -> Option<ast::BlockExpr> {
|
||||||
self.blocks().next()
|
self.blocks().next()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn else_branch(&self) -> Option<ElseBranch> {
|
pub fn else_branch(&self) -> Option<ElseBranch> {
|
||||||
let res = match self.blocks().nth(1) {
|
let res = match self.blocks().nth(1) {
|
||||||
Some(block) => ElseBranch::Block(block),
|
Some(block) => ElseBranch::Block(block),
|
||||||
|
|
Loading…
Reference in a new issue