From 6a7824787e231485d93c9a08a13d229cbcd14ded Mon Sep 17 00:00:00 2001 From: hafiz <20735482+ayazhafiz@users.noreply.github.com> Date: Wed, 20 May 2020 10:39:34 -0500 Subject: [PATCH] Preserve comments in empty statements (#4180) * Preserve comments in empty statements Closes #4018 * fixup! Preserve comments in empty statements --- src/stmt.rs | 4 ++++ src/visitor.rs | 22 +++++++++++++++++----- tests/source/issue-4018.rs | 13 +++++++++++++ tests/target/issue-4018.rs | 11 +++++++++++ 4 files changed, 45 insertions(+), 5 deletions(-) create mode 100644 tests/source/issue-4018.rs create mode 100644 tests/target/issue-4018.rs diff --git a/src/stmt.rs b/src/stmt.rs index 807cd9e5fad..0b3854425ea 100644 --- a/src/stmt.rs +++ b/src/stmt.rs @@ -52,6 +52,10 @@ impl<'a> Stmt<'a> { result } + pub(crate) fn is_empty(&self) -> bool { + matches!(self.inner.kind, ast::StmtKind::Empty) + } + fn is_last_expr(&self) -> bool { if !self.is_last { return false; diff --git a/src/visitor.rs b/src/visitor.rs index 523d4e9fea9..e7fa27aae0c 100644 --- a/src/visitor.rs +++ b/src/visitor.rs @@ -25,7 +25,7 @@ use crate::stmt::Stmt; use crate::syntux::session::ParseSess; use crate::utils::{ self, contains_skip, count_newlines, depr_skip_annotation, format_unsafety, inner_attributes, - last_line_width, mk_sp, ptr_vec_to_ref_vec, rewrite_ident, stmt_expr, + last_line_width, mk_sp, ptr_vec_to_ref_vec, rewrite_ident, starts_with_newline, stmt_expr, }; use crate::{ErrorKind, FormatReport, FormattingError}; @@ -117,10 +117,22 @@ impl<'b, 'a: 'b> FmtVisitor<'a> { self.parse_sess.span_to_debug_info(stmt.span()) ); - // https://github.com/rust-lang/rust/issues/63679. - let is_all_semicolons = - |snippet: &str| snippet.chars().all(|c| c.is_whitespace() || c == ';'); - if is_all_semicolons(&self.snippet(stmt.span())) { + if stmt.is_empty() { + // If the statement is empty, just skip over it. Before that, make sure any comment + // snippet preceding the semicolon is picked up. + let snippet = self.snippet(mk_sp(self.last_pos, stmt.span().lo())); + let original_starts_with_newline = snippet + .find(|c| c != ' ') + .map_or(false, |i| starts_with_newline(&snippet[i..])); + let snippet = snippet.trim(); + if !snippet.is_empty() { + if original_starts_with_newline { + self.push_str("\n"); + } + self.push_str(&self.block_indent.to_string(self.config)); + self.push_str(snippet); + } + self.last_pos = stmt.span().hi(); return; } diff --git a/tests/source/issue-4018.rs b/tests/source/issue-4018.rs new file mode 100644 index 00000000000..9a91dd9a306 --- /dev/null +++ b/tests/source/issue-4018.rs @@ -0,0 +1,13 @@ +fn main() { + ; + /* extra comment */ ; +} + +fn main() { + println!(""); + // comment 1 + // comment 2 + // comment 3 + // comment 4 + ; +} diff --git a/tests/target/issue-4018.rs b/tests/target/issue-4018.rs new file mode 100644 index 00000000000..cef3be06148 --- /dev/null +++ b/tests/target/issue-4018.rs @@ -0,0 +1,11 @@ +fn main() { + /* extra comment */ +} + +fn main() { + println!(""); + // comment 1 + // comment 2 + // comment 3 + // comment 4 +}