Skip to content

Commit 0dca70f

Browse files
committed
Merge branch 'matthew-mcallister-attrib-block-expr'
2 parents 314c912 + c516840 commit 0dca70f

File tree

5 files changed

+175
-49
lines changed

5 files changed

+175
-49
lines changed

src/closures.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,8 @@ pub fn rewrite_closure(
5050
if let ast::ExprKind::Block(ref block) = body.node {
5151
// The body of the closure is an empty block.
5252
if block.stmts.is_empty() && !block_contains_comment(block, context.codemap) {
53-
return Some(format!("{} {{}}", prefix));
53+
return body.rewrite(context, shape)
54+
.map(|s| format!("{} {}", prefix, s));
5455
}
5556

5657
let result = match fn_decl.output {
@@ -138,7 +139,7 @@ fn rewrite_closure_with_block(
138139
span: body.span,
139140
recovered: false,
140141
};
141-
let block = ::expr::rewrite_block_with_visitor(context, "", &block, shape, false)?;
142+
let block = ::expr::rewrite_block_with_visitor(context, "", &block, None, shape, false)?;
142143
Some(format!("{} {}", prefix, block))
143144
}
144145

@@ -291,7 +292,8 @@ pub fn rewrite_last_closure(
291292
if let ast::ExprKind::Closure(capture, movability, ref fn_decl, ref body, _) = expr.node {
292293
let body = match body.node {
293294
ast::ExprKind::Block(ref block)
294-
if !is_unsafe_block(block) && is_simple_block(block, context.codemap) =>
295+
if !is_unsafe_block(block)
296+
&& is_simple_block(block, Some(&body.attrs), context.codemap) =>
295297
{
296298
stmt_expr(&block.stmts[0]).unwrap_or(body)
297299
}

src/expr.rs

+93-38
Original file line numberDiff line numberDiff line change
@@ -115,16 +115,25 @@ pub fn format_expr(
115115
match expr_type {
116116
ExprType::Statement => {
117117
if is_unsafe_block(block) {
118-
block.rewrite(context, shape)
119-
} else if let rw @ Some(_) = rewrite_empty_block(context, block, shape) {
118+
rewrite_block(block, Some(&expr.attrs), context, shape)
119+
} else if let rw @ Some(_) =
120+
rewrite_empty_block(context, block, Some(&expr.attrs), "", shape)
121+
{
120122
// Rewrite block without trying to put it in a single line.
121123
rw
122124
} else {
123125
let prefix = block_prefix(context, block, shape)?;
124-
rewrite_block_with_visitor(context, &prefix, block, shape, true)
126+
rewrite_block_with_visitor(
127+
context,
128+
&prefix,
129+
block,
130+
Some(&expr.attrs),
131+
shape,
132+
true,
133+
)
125134
}
126135
}
127-
ExprType::SubExpression => block.rewrite(context, shape),
136+
ExprType::SubExpression => rewrite_block(block, Some(&expr.attrs), context, shape),
128137
}
129138
}
130139
ast::ExprKind::Match(ref cond, ref arms) => {
@@ -290,15 +299,22 @@ pub fn format_expr(
290299
Some(context.snippet(expr.span).to_owned())
291300
}
292301
ast::ExprKind::Catch(ref block) => {
293-
if let rw @ Some(_) = rewrite_single_line_block(context, "do catch ", block, shape) {
302+
if let rw @ Some(_) =
303+
rewrite_single_line_block(context, "do catch ", block, Some(&expr.attrs), shape)
304+
{
294305
rw
295306
} else {
296307
// 9 = `do catch `
297308
let budget = shape.width.checked_sub(9).unwrap_or(0);
298309
Some(format!(
299310
"{}{}",
300311
"do catch ",
301-
block.rewrite(context, Shape::legacy(budget, shape.indent))?
312+
rewrite_block(
313+
block,
314+
Some(&expr.attrs),
315+
context,
316+
Shape::legacy(budget, shape.indent)
317+
)?
302318
))
303319
}
304320
}
@@ -347,7 +363,7 @@ where
347363
let lhs_result = lhs.rewrite(context, lhs_shape)
348364
.map(|lhs_str| format!("{}{}", pp.prefix, lhs_str))?;
349365

350-
// Try to the both lhs and rhs on the same line.
366+
// Try to put both lhs and rhs on the same line.
351367
let rhs_orig_result = shape
352368
.offset_left(last_line_width(&lhs_result) + pp.infix.len())
353369
.and_then(|s| s.sub_width(pp.suffix.len()))
@@ -564,11 +580,17 @@ fn nop_block_collapse(block_str: Option<String>, budget: usize) -> Option<String
564580
fn rewrite_empty_block(
565581
context: &RewriteContext,
566582
block: &ast::Block,
583+
attrs: Option<&[ast::Attribute]>,
584+
prefix: &str,
567585
shape: Shape,
568586
) -> Option<String> {
587+
if attrs.map_or(false, |a| !inner_attributes(a).is_empty()) {
588+
return None;
589+
}
590+
569591
if block.stmts.is_empty() && !block_contains_comment(block, context.codemap) && shape.width >= 2
570592
{
571-
return Some("{}".to_owned());
593+
return Some(format!("{}{{}}", prefix));
572594
}
573595

574596
// If a block contains only a single-line comment, then leave it on one line.
@@ -579,7 +601,7 @@ fn rewrite_empty_block(
579601
if block.stmts.is_empty() && !comment_str.contains('\n') && !comment_str.starts_with("//")
580602
&& comment_str.len() + 4 <= shape.width
581603
{
582-
return Some(format!("{{ {} }}", comment_str));
604+
return Some(format!("{}{{ {} }}", prefix, comment_str));
583605
}
584606
}
585607

@@ -618,9 +640,10 @@ fn rewrite_single_line_block(
618640
context: &RewriteContext,
619641
prefix: &str,
620642
block: &ast::Block,
643+
attrs: Option<&[ast::Attribute]>,
621644
shape: Shape,
622645
) -> Option<String> {
623-
if is_simple_block(block, context.codemap) {
646+
if is_simple_block(block, attrs, context.codemap) {
624647
let expr_shape = shape.offset_left(last_line_width(prefix))?;
625648
let expr_str = block.stmts[0].rewrite(context, expr_shape)?;
626649
let result = format!("{}{{ {} }}", prefix, expr_str);
@@ -635,10 +658,11 @@ pub fn rewrite_block_with_visitor(
635658
context: &RewriteContext,
636659
prefix: &str,
637660
block: &ast::Block,
661+
attrs: Option<&[ast::Attribute]>,
638662
shape: Shape,
639663
has_braces: bool,
640664
) -> Option<String> {
641-
if let rw @ Some(_) = rewrite_empty_block(context, block, shape) {
665+
if let rw @ Some(_) = rewrite_empty_block(context, block, attrs, prefix, shape) {
642666
return rw;
643667
}
644668

@@ -654,31 +678,41 @@ pub fn rewrite_block_with_visitor(
654678
ast::BlockCheckMode::Default => visitor.last_pos = block.span.lo(),
655679
}
656680

657-
visitor.visit_block(block, None, has_braces);
681+
let inner_attrs = attrs.map(inner_attributes);
682+
visitor.visit_block(block, inner_attrs.as_ref().map(|a| &**a), has_braces);
658683
Some(format!("{}{}", prefix, visitor.buffer))
659684
}
660685

661686
impl Rewrite for ast::Block {
662687
fn rewrite(&self, context: &RewriteContext, shape: Shape) -> Option<String> {
663-
// shape.width is used only for the single line case: either the empty block `{}`,
664-
// or an unsafe expression `unsafe { e }`.
665-
if let rw @ Some(_) = rewrite_empty_block(context, self, shape) {
666-
return rw;
667-
}
688+
rewrite_block(self, None, context, shape)
689+
}
690+
}
668691

669-
let prefix = block_prefix(context, self, shape)?;
692+
fn rewrite_block(
693+
block: &ast::Block,
694+
attrs: Option<&[ast::Attribute]>,
695+
context: &RewriteContext,
696+
shape: Shape,
697+
) -> Option<String> {
698+
let prefix = block_prefix(context, block, shape)?;
670699

671-
let result = rewrite_block_with_visitor(context, &prefix, self, shape, true);
672-
if let Some(ref result_str) = result {
673-
if result_str.lines().count() <= 3 {
674-
if let rw @ Some(_) = rewrite_single_line_block(context, &prefix, self, shape) {
675-
return rw;
676-
}
700+
// shape.width is used only for the single line case: either the empty block `{}`,
701+
// or an unsafe expression `unsafe { e }`.
702+
if let rw @ Some(_) = rewrite_empty_block(context, block, attrs, &prefix, shape) {
703+
return rw;
704+
}
705+
706+
let result = rewrite_block_with_visitor(context, &prefix, block, attrs, shape, true);
707+
if let Some(ref result_str) = result {
708+
if result_str.lines().count() <= 3 {
709+
if let rw @ Some(_) = rewrite_single_line_block(context, &prefix, block, attrs, shape) {
710+
return rw;
677711
}
678712
}
679-
680-
result
681713
}
714+
715+
result
682716
}
683717

684718
impl Rewrite for ast::Stmt {
@@ -889,8 +923,8 @@ impl<'a> ControlFlow<'a> {
889923
let fixed_cost = self.keyword.len() + " { } else { }".len();
890924

891925
if let ast::ExprKind::Block(ref else_node) = else_block.node {
892-
if !is_simple_block(self.block, context.codemap)
893-
|| !is_simple_block(else_node, context.codemap)
926+
if !is_simple_block(self.block, None, context.codemap)
927+
|| !is_simple_block(else_node, None, context.codemap)
894928
|| pat_expr_str.contains('\n')
895929
{
896930
return None;
@@ -1123,7 +1157,7 @@ impl<'a> Rewrite for ControlFlow<'a> {
11231157
let mut block_context = context.clone();
11241158
block_context.is_if_else_block = self.else_block.is_some();
11251159
let block_str =
1126-
rewrite_block_with_visitor(&block_context, "", self.block, block_shape, true)?;
1160+
rewrite_block_with_visitor(&block_context, "", self.block, None, block_shape, true)?;
11271161

11281162
let mut result = format!("{}{}", cond_str, block_str);
11291163

@@ -1234,22 +1268,39 @@ pub fn block_contains_comment(block: &ast::Block, codemap: &CodeMap) -> bool {
12341268
contains_comment(&snippet)
12351269
}
12361270

1237-
// Checks that a block contains no statements, an expression and no comments.
1271+
// Checks that a block contains no statements, an expression and no comments or
1272+
// attributes.
12381273
// FIXME: incorrectly returns false when comment is contained completely within
12391274
// the expression.
1240-
pub fn is_simple_block(block: &ast::Block, codemap: &CodeMap) -> bool {
1275+
pub fn is_simple_block(
1276+
block: &ast::Block,
1277+
attrs: Option<&[ast::Attribute]>,
1278+
codemap: &CodeMap,
1279+
) -> bool {
12411280
(block.stmts.len() == 1 && stmt_is_expr(&block.stmts[0])
1242-
&& !block_contains_comment(block, codemap))
1281+
&& !block_contains_comment(block, codemap) && attrs.map_or(true, |a| a.is_empty()))
12431282
}
12441283

1245-
/// Checks whether a block contains at most one statement or expression, and no comments.
1246-
pub fn is_simple_block_stmt(block: &ast::Block, codemap: &CodeMap) -> bool {
1284+
/// Checks whether a block contains at most one statement or expression, and no
1285+
/// comments or attributes.
1286+
pub fn is_simple_block_stmt(
1287+
block: &ast::Block,
1288+
attrs: Option<&[ast::Attribute]>,
1289+
codemap: &CodeMap,
1290+
) -> bool {
12471291
block.stmts.len() <= 1 && !block_contains_comment(block, codemap)
1292+
&& attrs.map_or(true, |a| a.is_empty())
12481293
}
12491294

1250-
/// Checks whether a block contains no statements, expressions, or comments.
1251-
pub fn is_empty_block(block: &ast::Block, codemap: &CodeMap) -> bool {
1295+
/// Checks whether a block contains no statements, expressions, comments, or
1296+
/// inner attributes.
1297+
pub fn is_empty_block(
1298+
block: &ast::Block,
1299+
attrs: Option<&[ast::Attribute]>,
1300+
codemap: &CodeMap,
1301+
) -> bool {
12521302
block.stmts.is_empty() && !block_contains_comment(block, codemap)
1303+
&& attrs.map_or(true, |a| inner_attributes(a).is_empty())
12531304
}
12541305

12551306
pub fn stmt_is_expr(stmt: &ast::Stmt) -> bool {
@@ -1577,7 +1628,8 @@ fn rewrite_match_pattern(
15771628
fn flatten_arm_body<'a>(context: &'a RewriteContext, body: &'a ast::Expr) -> (bool, &'a ast::Expr) {
15781629
match body.node {
15791630
ast::ExprKind::Block(ref block)
1580-
if !is_unsafe_block(block) && is_simple_block(block, context.codemap) =>
1631+
if !is_unsafe_block(block)
1632+
&& is_simple_block(block, Some(&body.attrs), context.codemap) =>
15811633
{
15821634
if let ast::StmtKind::Expr(ref expr) = block.stmts[0].node {
15831635
(
@@ -1605,7 +1657,10 @@ fn rewrite_match_body(
16051657
) -> Option<String> {
16061658
let (extend, body) = flatten_arm_body(context, body);
16071659
let (is_block, is_empty_block) = if let ast::ExprKind::Block(ref block) = body.node {
1608-
(true, is_empty_block(block, context.codemap))
1660+
(
1661+
true,
1662+
is_empty_block(block, Some(&body.attrs), context.codemap),
1663+
)
16091664
} else {
16101665
(false, false)
16111666
};

src/items.rs

+12-5
Original file line numberDiff line numberDiff line change
@@ -301,6 +301,7 @@ impl<'a> FmtVisitor<'a> {
301301
fn_sig: &FnSig,
302302
span: Span,
303303
block: &ast::Block,
304+
inner_attrs: Option<&[ast::Attribute]>,
304305
) -> Option<String> {
305306
let context = self.get_context();
306307

@@ -329,7 +330,8 @@ impl<'a> FmtVisitor<'a> {
329330
result.push(' ');
330331
}
331332

332-
self.single_line_fn(&result, block).or_else(|| Some(result))
333+
self.single_line_fn(&result, block, inner_attrs)
334+
.or_else(|| Some(result))
333335
}
334336

335337
pub fn rewrite_required_fn(
@@ -360,20 +362,25 @@ impl<'a> FmtVisitor<'a> {
360362
Some(result)
361363
}
362364

363-
fn single_line_fn(&self, fn_str: &str, block: &ast::Block) -> Option<String> {
364-
if fn_str.contains('\n') {
365+
fn single_line_fn(
366+
&self,
367+
fn_str: &str,
368+
block: &ast::Block,
369+
inner_attrs: Option<&[ast::Attribute]>,
370+
) -> Option<String> {
371+
if fn_str.contains('\n') || inner_attrs.map_or(false, |a| !a.is_empty()) {
365372
return None;
366373
}
367374

368375
let codemap = self.get_context().codemap;
369376

370-
if self.config.empty_item_single_line() && is_empty_block(block, codemap)
377+
if self.config.empty_item_single_line() && is_empty_block(block, None, codemap)
371378
&& self.block_indent.width() + fn_str.len() + 2 <= self.config.max_width()
372379
{
373380
return Some(format!("{}{{}}", fn_str));
374381
}
375382

376-
if self.config.fn_single_line() && is_simple_block_stmt(block, codemap) {
383+
if self.config.fn_single_line() && is_simple_block_stmt(block, None, codemap) {
377384
let rewrite = {
378385
if let Some(stmt) = block.stmts.first() {
379386
match stmt_expr(stmt) {

src/visitor.rs

+7-3
Original file line numberDiff line numberDiff line change
@@ -268,6 +268,7 @@ impl<'b, 'a: 'b> FmtVisitor<'a> {
268268
&FnSig::from_fn_kind(&fk, generics, fd, defaultness),
269269
mk_sp(s.lo(), b.span.lo()),
270270
b,
271+
inner_attrs,
271272
)
272273
}
273274
visit::FnKind::Closure(_) => unreachable!(),
@@ -381,13 +382,14 @@ impl<'b, 'a: 'b> FmtVisitor<'a> {
381382
self.visit_static(&StaticParts::from_item(item));
382383
}
383384
ast::ItemKind::Fn(ref decl, unsafety, constness, abi, ref generics, ref body) => {
385+
let inner_attrs = inner_attributes(&item.attrs);
384386
self.visit_fn(
385387
visit::FnKind::ItemFn(item.ident, unsafety, constness, abi, &item.vis, body),
386388
generics,
387389
decl,
388390
item.span,
389391
ast::Defaultness::Final,
390-
Some(&item.attrs),
392+
Some(&inner_attrs),
391393
)
392394
}
393395
ast::ItemKind::Ty(ref ty, ref generics) => {
@@ -438,13 +440,14 @@ impl<'b, 'a: 'b> FmtVisitor<'a> {
438440
self.push_rewrite(ti.span, rewrite);
439441
}
440442
ast::TraitItemKind::Method(ref sig, Some(ref body)) => {
443+
let inner_attrs = inner_attributes(&ti.attrs);
441444
self.visit_fn(
442445
visit::FnKind::Method(ti.ident, sig, None, body),
443446
&ti.generics,
444447
&sig.decl,
445448
ti.span,
446449
ast::Defaultness::Final,
447-
Some(&ti.attrs),
450+
Some(&inner_attrs),
448451
);
449452
}
450453
ast::TraitItemKind::Type(ref type_param_bounds, ref type_default) => {
@@ -473,13 +476,14 @@ impl<'b, 'a: 'b> FmtVisitor<'a> {
473476

474477
match ii.node {
475478
ast::ImplItemKind::Method(ref sig, ref body) => {
479+
let inner_attrs = inner_attributes(&ii.attrs);
476480
self.visit_fn(
477481
visit::FnKind::Method(ii.ident, sig, Some(&ii.vis), body),
478482
&ii.generics,
479483
&sig.decl,
480484
ii.span,
481485
ii.defaultness,
482-
Some(&ii.attrs),
486+
Some(&inner_attrs),
483487
);
484488
}
485489
ast::ImplItemKind::Const(..) => self.visit_static(&StaticParts::from_impl_item(ii)),

0 commit comments

Comments
 (0)