9773: internal: Improve `extract_function` assist r=Veykril a=Veykril

- fix: It doesn't try to overwrite parts of selected comments any longer
- fix: It doesn't wrap tail expressions and return types in a result or option unnecessarily
- feat?: It now adds a `const` modifier to the created function if extract somethings from a const context 

Fixes #7840

Co-authored-by: Lukas Wirth <lukastw97@gmail.com>
This commit is contained in:
bors[bot] 2021-08-03 16:53:12 +00:00 committed by GitHub
commit 09ec0a15fa
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -8,6 +8,7 @@ use ide_db::{
search::{FileReference, ReferenceAccess, SearchScope}, search::{FileReference, ReferenceAccess, SearchScope},
RootDatabase, RootDatabase,
}; };
use itertools::Itertools;
use rustc_hash::FxHasher; use rustc_hash::FxHasher;
use stdx::format_to; use stdx::format_to;
use syntax::{ use syntax::{
@ -16,7 +17,7 @@ use syntax::{
edit::{AstNodeEdit, IndentLevel}, edit::{AstNodeEdit, IndentLevel},
AstNode, AstNode,
}, },
ted, match_ast, ted,
SyntaxKind::{self, COMMENT}, SyntaxKind::{self, COMMENT},
SyntaxNode, SyntaxToken, TextRange, TextSize, TokenAtOffset, WalkEvent, T, SyntaxNode, SyntaxToken, TextRange, TextSize, TokenAtOffset, WalkEvent, T,
}; };
@ -70,17 +71,17 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext) -> Option
syntax::NodeOrToken::Token(t) => t.parent()?, syntax::NodeOrToken::Token(t) => t.parent()?,
}; };
let body = extraction_target(&node, range)?; let body = extraction_target(&node, range)?;
let mods = body.analyze_container()?;
let (locals_used, has_await, self_param) = analyze_body(&ctx.sema, &body); let (locals_used, self_param) = body.analyze(&ctx.sema);
let anchor = if self_param.is_some() { Anchor::Method } else { Anchor::Freestanding }; let anchor = if self_param.is_some() { Anchor::Method } else { Anchor::Freestanding };
let insert_after = node_to_insert_after(&body, anchor)?; let insert_after = node_to_insert_after(&body, anchor)?;
let module = ctx.sema.scope(&insert_after).module()?; let module = ctx.sema.scope(&insert_after).module()?;
let ret_ty = body_return_ty(ctx, &body)?; let ret_ty = body.return_ty(ctx)?;
let ret_values = ret_values(ctx, &body, node.parent().as_ref().unwrap_or(&node)); let control_flow = body.external_control_flow(ctx)?;
let ret_values = body.ret_values(ctx, node.parent().as_ref().unwrap_or(&node));
let control_flow = external_control_flow(ctx, &body)?;
let target_range = body.text_range(); let target_range = body.text_range();
@ -89,18 +90,15 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext) -> Option
"Extract into function", "Extract into function",
target_range, target_range,
move |builder| { move |builder| {
let ret_values: Vec<_> = ret_values.collect(); let outliving_locals: Vec<_> = ret_values.collect();
if stdx::never!(!ret_values.is_empty() && !ret_ty.is_unit()) { if stdx::never!(!outliving_locals.is_empty() && !ret_ty.is_unit()) {
// We should not have variables that outlive body if we have expression block // We should not have variables that outlive body if we have expression block
stdx::never!();
return; return;
} }
let params = extracted_function_params(ctx, &body, locals_used.iter().copied()); let params = body.extracted_function_params(ctx, locals_used.iter().copied());
let insert_comma = body
.parent()
.and_then(ast::MatchArm::cast)
.map_or(false, |it| it.comma_token().is_none());
let fun = Function { let fun = Function {
name: "fun_name".to_string(), name: "fun_name".to_string(),
self_param, self_param,
@ -108,18 +106,16 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext) -> Option
control_flow, control_flow,
ret_ty, ret_ty,
body, body,
vars_defined_in_body_and_outlive: ret_values, outliving_locals,
mods,
}; };
let new_indent = IndentLevel::from_node(&insert_after); let new_indent = IndentLevel::from_node(&insert_after);
let old_indent = fun.body.indent_level(); let old_indent = fun.body.indent_level();
builder.replace(target_range, make_call(ctx, &fun, old_indent, has_await)); builder.replace(target_range, make_call(ctx, &fun, old_indent));
if insert_comma {
builder.insert(target_range.end(), ",");
}
let fn_def = format_function(ctx, module, &fun, old_indent, new_indent, has_await); let fn_def = format_function(ctx, module, &fun, old_indent, new_indent);
let insert_offset = insert_after.text_range().end(); let insert_offset = insert_after.text_range().end();
match ctx.config.snippet_cap { match ctx.config.snippet_cap {
Some(cap) => builder.insert_snippet(cap, insert_offset, fn_def), Some(cap) => builder.insert_snippet(cap, insert_offset, fn_def),
@ -128,6 +124,50 @@ pub(crate) fn extract_function(acc: &mut Assists, ctx: &AssistContext) -> Option
}, },
) )
} }
/// Try to guess what user wants to extract
///
/// We have basically have two cases:
/// * We want whole node, like `loop {}`, `2 + 2`, `{ let n = 1; }` exprs.
/// Then we can use `ast::Expr`
/// * We want a few statements for a block. E.g.
/// ```rust,no_run
/// fn foo() -> i32 {
/// let m = 1;
/// $0
/// let n = 2;
/// let k = 3;
/// k + n
/// $0
/// }
/// ```
///
fn extraction_target(node: &SyntaxNode, selection_range: TextRange) -> Option<FunctionBody> {
if let Some(stmt) = ast::Stmt::cast(node.clone()) {
return match stmt {
ast::Stmt::Item(_) => None,
ast::Stmt::ExprStmt(_) | ast::Stmt::LetStmt(_) => Some(FunctionBody::from_range(
node.parent().and_then(ast::BlockExpr::cast)?,
node.text_range(),
)),
};
}
let expr = ast::Expr::cast(node.clone())?;
// A node got selected fully
if node.text_range() == selection_range {
return FunctionBody::from_expr(expr.clone());
}
// Covering element returned the parent block of one or multiple statements that have been selected
if let ast::Expr::BlockExpr(block) = expr {
// Extract the full statements.
return Some(FunctionBody::from_range(block, selection_range));
}
node.ancestors().find_map(ast::Expr::cast).and_then(FunctionBody::from_expr)
}
#[derive(Debug)] #[derive(Debug)]
struct Function { struct Function {
name: String, name: String,
@ -136,7 +176,8 @@ struct Function {
control_flow: ControlFlow, control_flow: ControlFlow,
ret_ty: RetType, ret_ty: RetType,
body: FunctionBody, body: FunctionBody,
vars_defined_in_body_and_outlive: Vec<OutlivedLocal>, outliving_locals: Vec<OutlivedLocal>,
mods: Modifiers,
} }
#[derive(Debug)] #[derive(Debug)]
@ -175,6 +216,13 @@ enum Anchor {
#[derive(Debug)] #[derive(Debug)]
struct ControlFlow { struct ControlFlow {
kind: Option<FlowKind>, kind: Option<FlowKind>,
is_async: bool,
}
#[derive(Copy, Clone, Debug)]
struct Modifiers {
is_const: bool,
is_in_tail: bool,
} }
/// Control flow that is exported from extracted function /// Control flow that is exported from extracted function
@ -266,7 +314,7 @@ impl Function {
match &self.ret_ty { match &self.ret_ty {
RetType::Expr(ty) if ty.is_unit() => FunType::Unit, RetType::Expr(ty) if ty.is_unit() => FunType::Unit,
RetType::Expr(ty) => FunType::Single(ty.clone()), RetType::Expr(ty) => FunType::Single(ty.clone()),
RetType::Stmt => match self.vars_defined_in_body_and_outlive.as_slice() { RetType::Stmt => match self.outliving_locals.as_slice() {
[] => FunType::Unit, [] => FunType::Unit,
[var] => FunType::Single(var.local.ty(ctx.db())), [var] => FunType::Single(var.local.ty(ctx.db())),
vars => { vars => {
@ -324,6 +372,24 @@ impl Param {
} }
} }
impl TryKind {
fn of_ty(ty: hir::Type, ctx: &AssistContext) -> Option<TryKind> {
if ty.is_unknown() {
// We favour Result for `expr?`
return Some(TryKind::Result { ty });
}
let adt = ty.as_adt()?;
let name = adt.name(ctx.db());
// FIXME: use lang items to determine if it is std type or user defined
// E.g. if user happens to define type named `Option`, we would have false positive
match name.to_string().as_str() {
"Option" => Some(TryKind::Option),
"Result" => Some(TryKind::Result { ty }),
_ => None,
}
}
}
impl FlowKind { impl FlowKind {
fn make_result_handler(&self, expr: Option<ast::Expr>) -> ast::Expr { fn make_result_handler(&self, expr: Option<ast::Expr>) -> ast::Expr {
match self { match self {
@ -356,22 +422,6 @@ impl FlowKind {
} }
} }
fn try_kind_of_ty(ty: hir::Type, ctx: &AssistContext) -> Option<TryKind> {
if ty.is_unknown() {
// We favour Result for `expr?`
return Some(TryKind::Result { ty });
}
let adt = ty.as_adt()?;
let name = adt.name(ctx.db());
// FIXME: use lang items to determine if it is std type or user defined
// E.g. if user happens to define type named `Option`, we would have false positive
match name.to_string().as_str() {
"Option" => Some(TryKind::Option),
"Result" => Some(TryKind::Result { ty }),
_ => None,
}
}
impl FunctionBody { impl FunctionBody {
fn parent(&self) -> Option<SyntaxNode> { fn parent(&self) -> Option<SyntaxNode> {
match self { match self {
@ -389,8 +439,23 @@ impl FunctionBody {
} }
} }
fn from_range(parent: ast::BlockExpr, text_range: TextRange) -> FunctionBody { fn from_range(parent: ast::BlockExpr, selected: TextRange) -> FunctionBody {
Self::Span { parent, text_range } let mut text_range = parent
.statements()
.map(|stmt| stmt.syntax().text_range())
.filter(|&stmt| selected.intersect(stmt).filter(|it| !it.is_empty()).is_some())
.fold1(|acc, stmt| acc.cover(stmt));
if let Some(tail_range) = parent
.tail_expr()
.map(|it| it.syntax().text_range())
.filter(|&it| selected.intersect(it).is_some())
{
text_range = Some(match text_range {
Some(text_range) => text_range.cover(tail_range),
None => tail_range,
});
}
Self::Span { parent, text_range: text_range.unwrap_or(selected) }
} }
fn indent_level(&self) -> IndentLevel { fn indent_level(&self) -> IndentLevel {
@ -509,71 +574,17 @@ impl FunctionBody {
} }
} }
/// Try to guess what user wants to extract impl FunctionBody {
/// /// Analyzes a function body, returning the used local variables that are referenced in it as well as
/// We have basically have two cases: /// whether it contains an await expression.
/// * We want whole node, like `loop {}`, `2 + 2`, `{ let n = 1; }` exprs. fn analyze(
/// Then we can use `ast::Expr` &self,
/// * We want a few statements for a block. E.g.
/// ```rust,no_run
/// fn foo() -> i32 {
/// let m = 1;
/// $0
/// let n = 2;
/// let k = 3;
/// k + n
/// $0
/// }
/// ```
///
fn extraction_target(node: &SyntaxNode, selection_range: TextRange) -> Option<FunctionBody> {
if let Some(stmt) = ast::Stmt::cast(node.clone()) {
return match stmt {
ast::Stmt::Item(_) => None,
ast::Stmt::ExprStmt(_) | ast::Stmt::LetStmt(_) => Some(FunctionBody::from_range(
node.parent().and_then(ast::BlockExpr::cast)?,
node.text_range(),
)),
};
}
let expr = ast::Expr::cast(node.clone())?;
// A node got selected fully
if node.text_range() == selection_range {
return FunctionBody::from_expr(expr.clone());
}
// Covering element returned the parent block of one or multiple statements that have been selected
if let ast::Expr::BlockExpr(block) = expr {
// Extract the full statements.
let mut statements_range = block
.statements()
.filter(|stmt| selection_range.intersect(stmt.syntax().text_range()).is_some())
.fold(selection_range, |acc, stmt| acc.cover(stmt.syntax().text_range()));
if let Some(e) = block
.tail_expr()
.filter(|it| selection_range.intersect(it.syntax().text_range()).is_some())
{
statements_range = statements_range.cover(e.syntax().text_range());
}
return Some(FunctionBody::from_range(block, statements_range));
}
node.ancestors().find_map(ast::Expr::cast).and_then(FunctionBody::from_expr)
}
/// Analyzes a function body, returning the used local variables that are referenced in it as well as
/// whether it contains an await expression.
fn analyze_body(
sema: &Semantics<RootDatabase>, sema: &Semantics<RootDatabase>,
body: &FunctionBody, ) -> (FxIndexSet<Local>, Option<ast::SelfParam>) {
) -> (FxIndexSet<Local>, bool, Option<ast::SelfParam>) {
// FIXME: currently usages inside macros are not found // FIXME: currently usages inside macros are not found
let mut has_await = false;
let mut self_param = None; let mut self_param = None;
let mut res = FxIndexSet::default(); let mut res = FxIndexSet::default();
body.walk_expr(&mut |expr| { self.walk_expr(&mut |expr| {
has_await |= matches!(expr, ast::Expr::AwaitExpr(_));
let name_ref = match expr { let name_ref = match expr {
ast::Expr::PathExpr(path_expr) => { ast::Expr::PathExpr(path_expr) => {
path_expr.path().and_then(|it| it.as_single_name_ref()) path_expr.path().and_then(|it| it.as_single_name_ref())
@ -586,7 +597,6 @@ fn analyze_body(
| NameRefClass::FieldShorthand { local_ref, field_ref: _ }, | NameRefClass::FieldShorthand { local_ref, field_ref: _ },
) = NameRefClass::classify(sema, &name_ref) ) = NameRefClass::classify(sema, &name_ref)
{ {
res.insert(local_ref);
if local_ref.is_self(sema.db) { if local_ref.is_self(sema.db) {
match local_ref.source(sema.db).value { match local_ref.source(sema.db).value {
Either::Right(it) => { Either::Right(it) => {
@ -596,28 +606,187 @@ fn analyze_body(
); );
} }
Either::Left(_) => { Either::Left(_) => {
stdx::never!("Local::is_self returned true, but source is IdentPat"); stdx::never!(
"Local::is_self returned true, but source is IdentPat"
);
} }
} }
} else {
res.insert(local_ref);
} }
} }
} }
}); });
(res, has_await, self_param) (res, self_param)
} }
/// find variables that should be extracted as params fn analyze_container(&self) -> Option<Modifiers> {
/// let mut is_const = false;
/// Computes additional info that affects param type and mutability let container_expr = self.parent()?.ancestors().find_map(|it| {
fn extracted_function_params( // double Option as we want to short circuit
let res = match_ast! {
match it {
ast::ClosureExpr(closure) => closure.body(),
ast::EffectExpr(effect) => {
is_const = effect.const_token().is_some();
effect.block_expr().map(ast::Expr::BlockExpr)
},
ast::Fn(fn_) => {
is_const = fn_.const_token().is_some();
fn_.body().map(ast::Expr::BlockExpr)
},
ast::Static(statik) => {
is_const = true;
statik.body()
},
ast::ConstArg(ca) => {
is_const = true;
ca.expr()
},
ast::Const(konst) => {
is_const = true;
konst.body()
},
ast::ConstParam(cp) => {
is_const = true;
cp.default_val()
},
ast::ConstBlockPat(cbp) => {
is_const = true;
cbp.block_expr().map(ast::Expr::BlockExpr)
},
ast::Variant(__) => None,
ast::Meta(__) => None,
_ => return None,
}
};
Some(res)
})??;
let container_tail = match container_expr {
ast::Expr::BlockExpr(block) => block.tail_expr(),
expr => Some(expr),
};
let is_in_tail =
container_tail.zip(self.tail_expr()).map_or(false, |(container_tail, body_tail)| {
container_tail.syntax().text_range().contains_range(body_tail.syntax().text_range())
});
Some(Modifiers { is_in_tail, is_const })
}
fn return_ty(&self, ctx: &AssistContext) -> Option<RetType> {
match self.tail_expr() {
Some(expr) => ctx.sema.type_of_expr(&expr).map(TypeInfo::original).map(RetType::Expr),
None => Some(RetType::Stmt),
}
}
/// Local variables defined inside `body` that are accessed outside of it
fn ret_values<'a>(
&self,
ctx: &'a AssistContext,
parent: &SyntaxNode,
) -> impl Iterator<Item = OutlivedLocal> + 'a {
let parent = parent.clone();
let range = self.text_range();
locals_defined_in_body(&ctx.sema, self)
.into_iter()
.filter_map(move |local| local_outlives_body(ctx, range, local, &parent))
}
/// Analyses the function body for external control flow.
fn external_control_flow(&self, ctx: &AssistContext) -> Option<ControlFlow> {
let mut ret_expr = None;
let mut try_expr = None;
let mut break_expr = None;
let mut continue_expr = None;
let mut is_async = false;
let mut loop_depth = 0;
self.preorder_expr(&mut |expr| {
let expr = match expr {
WalkEvent::Enter(e) => e,
WalkEvent::Leave(
ast::Expr::LoopExpr(_) | ast::Expr::ForExpr(_) | ast::Expr::WhileExpr(_),
) => {
loop_depth -= 1;
return false;
}
WalkEvent::Leave(_) => return false,
};
match expr {
ast::Expr::LoopExpr(_) | ast::Expr::ForExpr(_) | ast::Expr::WhileExpr(_) => {
loop_depth += 1;
}
ast::Expr::ReturnExpr(it) => {
ret_expr = Some(it);
}
ast::Expr::TryExpr(it) => {
try_expr = Some(it);
}
ast::Expr::BreakExpr(it) if loop_depth == 0 => {
break_expr = Some(it);
}
ast::Expr::ContinueExpr(it) if loop_depth == 0 => {
continue_expr = Some(it);
}
ast::Expr::AwaitExpr(_) => is_async = true,
_ => {}
}
false
});
let kind = match (try_expr, ret_expr, break_expr, continue_expr) {
(Some(e), None, None, None) => {
let func = e.syntax().ancestors().find_map(ast::Fn::cast)?;
let def = ctx.sema.to_def(&func)?;
let ret_ty = def.ret_type(ctx.db());
let kind = TryKind::of_ty(ret_ty, ctx)?;
Some(FlowKind::Try { kind })
}
(Some(_), Some(r), None, None) => match r.expr() {
Some(expr) => {
if let Some(kind) = expr_err_kind(&expr, ctx) {
Some(FlowKind::TryReturn { expr, kind })
} else {
cov_mark::hit!(external_control_flow_try_and_return_non_err);
return None;
}
}
None => return None,
},
(Some(_), _, _, _) => {
cov_mark::hit!(external_control_flow_try_and_bc);
return None;
}
(None, Some(r), None, None) => Some(FlowKind::Return(r.expr())),
(None, Some(_), _, _) => {
cov_mark::hit!(external_control_flow_return_and_bc);
return None;
}
(None, None, Some(_), Some(_)) => {
cov_mark::hit!(external_control_flow_break_and_continue);
return None;
}
(None, None, Some(b), None) => Some(FlowKind::Break(b.expr())),
(None, None, None, Some(_)) => Some(FlowKind::Continue),
(None, None, None, None) => None,
};
Some(ControlFlow { kind, is_async })
}
/// find variables that should be extracted as params
///
/// Computes additional info that affects param type and mutability
fn extracted_function_params(
&self,
ctx: &AssistContext, ctx: &AssistContext,
body: &FunctionBody,
locals: impl Iterator<Item = Local>, locals: impl Iterator<Item = Local>,
) -> Vec<Param> { ) -> Vec<Param> {
locals locals
.filter(|local| !local.is_self(ctx.db()))
.map(|local| (local, local.source(ctx.db()))) .map(|local| (local, local.source(ctx.db())))
.filter(|(_, src)| is_defined_outside_of_body(ctx, body, src)) .filter(|(_, src)| is_defined_outside_of_body(ctx, self, src))
.filter_map(|(local, src)| { .filter_map(|(local, src)| {
if src.value.is_left() { if src.value.is_left() {
Some(local) Some(local)
@ -633,16 +802,17 @@ fn extracted_function_params(
Param { Param {
var, var,
ty, ty,
has_usages_afterwards: has_usages_after_body(&usages, body), has_usages_afterwards: self.has_usages_after_body(&usages),
has_mut_inside_body: has_exclusive_usages(ctx, &usages, body), has_mut_inside_body: has_exclusive_usages(ctx, &usages, self),
is_copy, is_copy,
} }
}) })
.collect() .collect()
} }
fn has_usages_after_body(usages: &LocalUsages, body: &FunctionBody) -> bool { fn has_usages_after_body(&self, usages: &LocalUsages) -> bool {
usages.iter().any(|reference| body.precedes_range(reference.range)) usages.iter().any(|reference| self.precedes_range(reference.range))
}
} }
/// checks if relevant var is used with `&mut` access inside body /// checks if relevant var is used with `&mut` access inside body
@ -795,19 +965,6 @@ fn locals_defined_in_body(
res res
} }
/// Local variables defined inside `body` that are accessed outside of it
fn ret_values<'a>(
ctx: &'a AssistContext,
body: &FunctionBody,
parent: &SyntaxNode,
) -> impl Iterator<Item = OutlivedLocal> + 'a {
let parent = parent.clone();
let range = body.text_range();
locals_defined_in_body(&ctx.sema, body)
.into_iter()
.filter_map(move |local| local_outlives_body(ctx, range, local, &parent))
}
/// Returns usage details if local variable is used after(outside of) body /// Returns usage details if local variable is used after(outside of) body
fn local_outlives_body( fn local_outlives_body(
ctx: &AssistContext, ctx: &AssistContext,
@ -850,95 +1007,6 @@ fn either_syntax(value: &Either<ast::IdentPat, ast::SelfParam>) -> &SyntaxNode {
} }
} }
fn body_return_ty(ctx: &AssistContext, body: &FunctionBody) -> Option<RetType> {
match body.tail_expr() {
Some(expr) => ctx.sema.type_of_expr(&expr).map(TypeInfo::original).map(RetType::Expr),
None => Some(RetType::Stmt),
}
}
/// Analyses the function body for external control flow.
fn external_control_flow(ctx: &AssistContext, body: &FunctionBody) -> Option<ControlFlow> {
let mut ret_expr = None;
let mut try_expr = None;
let mut break_expr = None;
let mut continue_expr = None;
let mut loop_depth = 0;
body.preorder_expr(&mut |expr| {
let expr = match expr {
WalkEvent::Enter(e) => e,
WalkEvent::Leave(
ast::Expr::LoopExpr(_) | ast::Expr::ForExpr(_) | ast::Expr::WhileExpr(_),
) => {
loop_depth -= 1;
return false;
}
WalkEvent::Leave(_) => return false,
};
match expr {
ast::Expr::LoopExpr(_) | ast::Expr::ForExpr(_) | ast::Expr::WhileExpr(_) => {
loop_depth += 1;
}
ast::Expr::ReturnExpr(it) => {
ret_expr = Some(it);
}
ast::Expr::TryExpr(it) => {
try_expr = Some(it);
}
ast::Expr::BreakExpr(it) if loop_depth == 0 => {
break_expr = Some(it);
}
ast::Expr::ContinueExpr(it) if loop_depth == 0 => {
continue_expr = Some(it);
}
_ => {}
}
false
});
let kind = match (try_expr, ret_expr, break_expr, continue_expr) {
(Some(e), None, None, None) => {
let func = e.syntax().ancestors().find_map(ast::Fn::cast)?;
let def = ctx.sema.to_def(&func)?;
let ret_ty = def.ret_type(ctx.db());
let kind = try_kind_of_ty(ret_ty, ctx)?;
Some(FlowKind::Try { kind })
}
(Some(_), Some(r), None, None) => match r.expr() {
Some(expr) => {
if let Some(kind) = expr_err_kind(&expr, ctx) {
Some(FlowKind::TryReturn { expr, kind })
} else {
cov_mark::hit!(external_control_flow_try_and_return_non_err);
return None;
}
}
None => return None,
},
(Some(_), _, _, _) => {
cov_mark::hit!(external_control_flow_try_and_bc);
return None;
}
(None, Some(r), None, None) => Some(FlowKind::Return(r.expr())),
(None, Some(_), _, _) => {
cov_mark::hit!(external_control_flow_return_and_bc);
return None;
}
(None, None, Some(_), Some(_)) => {
cov_mark::hit!(external_control_flow_break_and_continue);
return None;
}
(None, None, Some(b), None) => Some(FlowKind::Break(b.expr())),
(None, None, None, Some(_)) => Some(FlowKind::Continue),
(None, None, None, None) => None,
};
Some(ControlFlow { kind })
}
/// Checks is expr is `Err(_)` or `None` /// Checks is expr is `Err(_)` or `None`
fn expr_err_kind(expr: &ast::Expr, ctx: &AssistContext) -> Option<TryKind> { fn expr_err_kind(expr: &ast::Expr, ctx: &AssistContext) -> Option<TryKind> {
let func_name = match expr { let func_name = match expr {
@ -991,12 +1059,7 @@ fn node_to_insert_after(body: &FunctionBody, anchor: Anchor) -> Option<SyntaxNod
last_ancestor last_ancestor
} }
fn make_call( fn make_call(ctx: &AssistContext, fun: &Function, indent: IndentLevel) -> String {
ctx: &AssistContext,
fun: &Function,
indent: IndentLevel,
body_contains_await: bool,
) -> String {
let ret_ty = fun.return_type(ctx); let ret_ty = fun.return_type(ctx);
let args = fun.params.iter().map(|param| param.to_arg(ctx)); let args = fun.params.iter().map(|param| param.to_arg(ctx));
@ -1013,8 +1076,10 @@ fn make_call(
let expr = handler.make_call_expr(call_expr).indent(indent); let expr = handler.make_call_expr(call_expr).indent(indent);
let mut_modifier = |var: &OutlivedLocal| if var.mut_usage_outside_body { "mut " } else { "" };
let mut buf = String::new(); let mut buf = String::new();
match fun.vars_defined_in_body_and_outlive.as_slice() { match fun.outliving_locals.as_slice() {
[] => {} [] => {}
[var] => { [var] => {
format_to!(buf, "let {}{} = ", mut_modifier(var), var.local.name(ctx.db()).unwrap()) format_to!(buf, "let {}{} = ", mut_modifier(var), var.local.name(ctx.db()).unwrap())
@ -1028,20 +1093,18 @@ fn make_call(
buf.push_str(") = "); buf.push_str(") = ");
} }
} }
fn mut_modifier(var: &OutlivedLocal) -> &'static str {
if var.mut_usage_outside_body {
"mut "
} else {
""
}
}
format_to!(buf, "{}", expr); format_to!(buf, "{}", expr);
if body_contains_await { if fun.control_flow.is_async {
buf.push_str(".await"); buf.push_str(".await");
} }
if fun.ret_ty.is_unit() let insert_comma = fun
&& (!fun.vars_defined_in_body_and_outlive.is_empty() || !expr.is_block_like()) .body
{ .parent()
.and_then(ast::MatchArm::cast)
.map_or(false, |it| it.comma_token().is_none());
if insert_comma {
buf.push(',');
} else if fun.ret_ty.is_unit() && (!fun.outliving_locals.is_empty() || !expr.is_block_like()) {
buf.push(';'); buf.push(';');
} }
buf buf
@ -1171,16 +1234,32 @@ fn format_function(
fun: &Function, fun: &Function,
old_indent: IndentLevel, old_indent: IndentLevel,
new_indent: IndentLevel, new_indent: IndentLevel,
body_contains_await: bool,
) -> String { ) -> String {
let mut fn_def = String::new(); let mut fn_def = String::new();
let params = make_param_list(ctx, module, fun); let params = fun.make_param_list(ctx, module);
let ret_ty = make_ret_ty(ctx, module, fun); let ret_ty = fun.make_ret_ty(ctx, module);
let body = make_body(ctx, old_indent, new_indent, fun); let body = make_body(ctx, old_indent, new_indent, fun);
let async_kw = if body_contains_await { "async " } else { "" }; let const_kw = if fun.mods.is_const { "const " } else { "" };
let async_kw = if fun.control_flow.is_async { "async " } else { "" };
match ctx.config.snippet_cap { match ctx.config.snippet_cap {
Some(_) => format_to!(fn_def, "\n\n{}{}fn $0{}{}", new_indent, async_kw, fun.name, params), Some(_) => format_to!(
None => format_to!(fn_def, "\n\n{}{}fn {}{}", new_indent, async_kw, fun.name, params), fn_def,
"\n\n{}{}{}fn $0{}{}",
new_indent,
const_kw,
async_kw,
fun.name,
params
),
None => format_to!(
fn_def,
"\n\n{}{}{}fn {}{}",
new_indent,
const_kw,
async_kw,
fun.name,
params
),
} }
if let Some(ret_ty) = ret_ty { if let Some(ret_ty) = ret_ty {
format_to!(fn_def, " {}", ret_ty); format_to!(fn_def, " {}", ret_ty);
@ -1190,38 +1269,20 @@ fn format_function(
fn_def fn_def
} }
fn make_param_list(ctx: &AssistContext, module: hir::Module, fun: &Function) -> ast::ParamList { impl Function {
let self_param = fun.self_param.clone(); fn make_param_list(&self, ctx: &AssistContext, module: hir::Module) -> ast::ParamList {
let params = fun.params.iter().map(|param| param.to_param(ctx, module)); let self_param = self.self_param.clone();
let params = self.params.iter().map(|param| param.to_param(ctx, module));
make::param_list(self_param, params) make::param_list(self_param, params)
} }
impl FunType { fn make_ret_ty(&self, ctx: &AssistContext, module: hir::Module) -> Option<ast::RetType> {
fn make_ty(&self, ctx: &AssistContext, module: hir::Module) -> ast::Type { let fun_ty = self.return_type(ctx);
match self { let handler = if self.mods.is_in_tail {
FunType::Unit => make::ty_unit(), FlowHandler::None
FunType::Single(ty) => make_ty(ty, ctx, module), } else {
FunType::Tuple(types) => match types.as_slice() { FlowHandler::from_ret_ty(self, &fun_ty)
[] => { };
stdx::never!("tuple type with 0 elements");
make::ty_unit()
}
[ty] => {
stdx::never!("tuple type with 1 element");
make_ty(ty, ctx, module)
}
types => {
let types = types.iter().map(|ty| make_ty(ty, ctx, module));
make::ty_tuple(types)
}
},
}
}
}
fn make_ret_ty(ctx: &AssistContext, module: hir::Module, fun: &Function) -> Option<ast::RetType> {
let fun_ty = fun.return_type(ctx);
let handler = FlowHandler::from_ret_ty(fun, &fun_ty);
let ret_ty = match &handler { let ret_ty = match &handler {
FlowHandler::None => { FlowHandler::None => {
if matches!(fun_ty, FunType::Unit) { if matches!(fun_ty, FunType::Unit) {
@ -1250,12 +1311,38 @@ fn make_ret_ty(ctx: &AssistContext, module: hir::Module, fun: &Function) -> Opti
} }
FlowHandler::MatchOption { .. } => make::ext::ty_option(fun_ty.make_ty(ctx, module)), FlowHandler::MatchOption { .. } => make::ext::ty_option(fun_ty.make_ty(ctx, module)),
FlowHandler::MatchResult { err } => { FlowHandler::MatchResult { err } => {
let handler_ty = let handler_ty = err
err.expr_ty(ctx).map(|ty| make_ty(&ty, ctx, module)).unwrap_or_else(make::ty_unit); .expr_ty(ctx)
.map(|ty| make_ty(&ty, ctx, module))
.unwrap_or_else(make::ty_unit);
make::ext::ty_result(fun_ty.make_ty(ctx, module), handler_ty) make::ext::ty_result(fun_ty.make_ty(ctx, module), handler_ty)
} }
}; };
Some(make::ret_type(ret_ty)) Some(make::ret_type(ret_ty))
}
}
impl FunType {
fn make_ty(&self, ctx: &AssistContext, module: hir::Module) -> ast::Type {
match self {
FunType::Unit => make::ty_unit(),
FunType::Single(ty) => make_ty(ty, ctx, module),
FunType::Tuple(types) => match types.as_slice() {
[] => {
stdx::never!("tuple type with 0 elements");
make::ty_unit()
}
[ty] => {
stdx::never!("tuple type with 1 element");
make_ty(ty, ctx, module)
}
types => {
let types = types.iter().map(|ty| make_ty(ty, ctx, module));
make::ty_tuple(types)
}
},
}
}
} }
fn make_body( fn make_body(
@ -1265,7 +1352,11 @@ fn make_body(
fun: &Function, fun: &Function,
) -> ast::BlockExpr { ) -> ast::BlockExpr {
let ret_ty = fun.return_type(ctx); let ret_ty = fun.return_type(ctx);
let handler = FlowHandler::from_ret_ty(fun, &ret_ty); let handler = if fun.mods.is_in_tail {
FlowHandler::None
} else {
FlowHandler::from_ret_ty(fun, &ret_ty)
};
let block = match &fun.body { let block = match &fun.body {
FunctionBody::Expr(expr) => { FunctionBody::Expr(expr) => {
let expr = rewrite_body_segment(ctx, &fun.params, &handler, expr.syntax()); let expr = rewrite_body_segment(ctx, &fun.params, &handler, expr.syntax());
@ -1301,7 +1392,7 @@ fn make_body(
}; };
if tail_expr.is_none() { if tail_expr.is_none() {
match fun.vars_defined_in_body_and_outlive.as_slice() { match fun.outliving_locals.as_slice() {
[] => {} [] => {}
[var] => { [var] => {
tail_expr = Some(path_expr_from_local(ctx, var.local)); tail_expr = Some(path_expr_from_local(ctx, var.local));
@ -3230,8 +3321,7 @@ fn $0fun_name(n: i32) -> bool {
r#" r#"
fn foo() { fn foo() {
loop { loop {
let n = 1; let n = 1;$0
$0
let k = 1; let k = 1;
loop { loop {
return; return;
@ -3435,8 +3525,7 @@ fn $0fun_name() -> Option<i32> {
r#" r#"
fn foo() -> i64 { fn foo() -> i64 {
loop { loop {
let n = 1; let n = 1;$0
$0
let k = 1; let k = 1;
if k == 42 { if k == 42 {
break 3; break 3;
@ -3830,6 +3919,95 @@ fn main() {
fn $0fun_name() -> i32 { fn $0fun_name() -> i32 {
100 100
} }
"#,
);
}
#[test]
fn extract_does_not_tear_comments_apart() {
check_assist(
extract_function,
r#"
fn foo() {
/*$0*/
foo();
foo();
/*$0*/
}
"#,
r#"
fn foo() {
/**/
fun_name();
/**/
}
fn $0fun_name() {
foo();
foo();
}
"#,
);
}
#[test]
fn extract_does_not_wrap_res_in_res() {
check_assist(
extract_function,
r#"
//- minicore: result
fn foo() -> Result<(), i64> {
$0Result::<i32, i64>::Ok(0)?;
Ok(())$0
}
"#,
r#"
fn foo() -> Result<(), i64> {
fun_name()?
}
fn $0fun_name() -> Result<(), i64> {
Result::<i32, i64>::Ok(0)?;
Ok(())
}
"#,
);
}
#[test]
fn extract_knows_const() {
check_assist(
extract_function,
r#"
const fn foo() {
$0()$0
}
"#,
r#"
const fn foo() {
fun_name();
}
const fn $0fun_name() {
()
}
"#,
);
check_assist(
extract_function,
r#"
const FOO: () = {
$0()$0
};
"#,
r#"
const FOO: () = {
fun_name();
};
const fn $0fun_name() {
()
}
"#, "#,
); );
} }