mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-14 22:24:14 +00:00
Merge #9773
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:
commit
09ec0a15fa
1 changed files with 519 additions and 341 deletions
|
@ -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() {
|
||||||
|
()
|
||||||
|
}
|
||||||
"#,
|
"#,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue