use crate::utils::{snippet_opt, span_help_and_lint, span_lint_and_sugg}; use crate::utils::sym; use rustc::lint::{EarlyContext, EarlyLintPass, LintArray, LintPass}; use rustc::{declare_lint_pass, declare_tool_lint}; use rustc_errors::Applicability; use syntax::ast; use syntax::source_map::Span; use syntax::tokenstream::TokenStream; declare_clippy_lint! { /// **What it does:** Checks for usage of dbg!() macro. /// /// **Why is this bad?** `dbg!` macro is intended as a debugging tool. It /// should not be in version control. /// /// **Known problems:** None. /// /// **Example:** /// ```rust,ignore /// // Bad /// dbg!(true) /// /// // Good /// true /// ``` pub DBG_MACRO, restriction, "`dbg!` macro is intended as a debugging tool" } declare_lint_pass!(DbgMacro => [DBG_MACRO]); impl EarlyLintPass for DbgMacro { fn check_mac(&mut self, cx: &EarlyContext<'_>, mac: &ast::Mac) { if mac.node.path == *sym::dbg { if let Some(sugg) = tts_span(mac.node.tts.clone()).and_then(|span| snippet_opt(cx, span)) { span_lint_and_sugg( cx, DBG_MACRO, mac.span, "`dbg!` macro is intended as a debugging tool", "ensure to avoid having uses of it in version control", sugg, Applicability::MaybeIncorrect, ); } else { span_help_and_lint( cx, DBG_MACRO, mac.span, "`dbg!` macro is intended as a debugging tool", "ensure to avoid having uses of it in version control", ); } } } } // Get span enclosing entire the token stream. fn tts_span(tts: TokenStream) -> Option { let mut cursor = tts.into_trees(); let first = cursor.next()?.span(); let span = match cursor.last() { Some(tree) => first.to(tree.span()), None => first, }; Some(span) }