Skip to content

Commit 5889312

Browse files
committed
Merge commit '3e4179766bcecd712824da04356621b8df012ea4' into sync-from-clippy
2 parents d95d4f0 + 3e41797 commit 5889312

35 files changed

+518
-126
lines changed

src/tools/clippy/CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -2079,6 +2079,7 @@ Released 2018-09-13
20792079
[`missing_docs_in_private_items`]: https://rust-lang.github.io/rust-clippy/master/index.html#missing_docs_in_private_items
20802080
[`missing_errors_doc`]: https://rust-lang.github.io/rust-clippy/master/index.html#missing_errors_doc
20812081
[`missing_inline_in_public_items`]: https://rust-lang.github.io/rust-clippy/master/index.html#missing_inline_in_public_items
2082+
[`missing_panics_doc`]: https://rust-lang.github.io/rust-clippy/master/index.html#missing_panics_doc
20822083
[`missing_safety_doc`]: https://rust-lang.github.io/rust-clippy/master/index.html#missing_safety_doc
20832084
[`mistyped_literal_suffixes`]: https://rust-lang.github.io/rust-clippy/master/index.html#mistyped_literal_suffixes
20842085
[`mixed_case_hex_literals`]: https://rust-lang.github.io/rust-clippy/master/index.html#mixed_case_hex_literals

src/tools/clippy/CONTRIBUTING.md

+5-4
Original file line numberDiff line numberDiff line change
@@ -46,11 +46,12 @@ first read the [Basics docs](doc/basics.md).**
4646

4747
### Finding something to fix/improve
4848

49-
All issues on Clippy are mentored, if you want help with a bug just ask
50-
@Manishearth, @flip1995, @phansch or @yaahc.
49+
All issues on Clippy are mentored, if you want help simply ask @Manishearth, @flip1995, @phansch
50+
or @llogiq directly by mentioning them in the issue or over on [Zulip]. This list may be out of date.
51+
All currently active mentors can be found [here](https://github.com/rust-lang/highfive/blob/master/highfive/configs/rust-lang/rust-clippy.json#L3)
5152

52-
Some issues are easier than others. The [`good-first-issue`] label can be used to find the easy issues.
53-
If you want to work on an issue, please leave a comment so that we can assign it to you!
53+
Some issues are easier than others. The [`good-first-issue`] label can be used to find the easy
54+
issues. You can use `@rustbot claim` to assign the issue to yourself.
5455

5556
There are also some abandoned PRs, marked with [`S-inactive-closed`].
5657
Pretty often these PRs are nearly completed and just need some extra steps

src/tools/clippy/clippy_dev/src/bless.rs

+3
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,9 @@ static CLIPPY_BUILD_TIME: SyncLazy<Option<std::time::SystemTime>> = SyncLazy::ne
2424
fs::metadata(path).ok()?.modified().ok()
2525
});
2626

27+
/// # Panics
28+
///
29+
/// Panics if the path to a test file is broken
2730
pub fn bless(ignore_timestamp: bool) {
2831
let test_suite_dirs = [
2932
clippy_project_root().join("tests").join("ui"),

src/tools/clippy/clippy_dev/src/fmt.rs

+15-8
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use walkdir::WalkDir;
88

99
#[derive(Debug)]
1010
pub enum CliError {
11-
CommandFailed(String),
11+
CommandFailed(String, String),
1212
IoError(io::Error),
1313
RustfmtNotInstalled,
1414
WalkDirError(walkdir::Error),
@@ -75,8 +75,8 @@ pub fn run(check: bool, verbose: bool) {
7575

7676
fn output_err(err: CliError) {
7777
match err {
78-
CliError::CommandFailed(command) => {
79-
eprintln!("error: A command failed! `{}`", command);
78+
CliError::CommandFailed(command, stderr) => {
79+
eprintln!("error: A command failed! `{}`\nstderr: {}", command, stderr);
8080
},
8181
CliError::IoError(err) => {
8282
eprintln!("error: {}", err);
@@ -136,12 +136,16 @@ fn exec(
136136
println!("{}", format_command(&program, &dir, args));
137137
}
138138

139-
let mut child = Command::new(&program).current_dir(&dir).args(args.iter()).spawn()?;
140-
let code = child.wait()?;
141-
let success = code.success();
139+
let child = Command::new(&program).current_dir(&dir).args(args.iter()).spawn()?;
140+
let output = child.wait_with_output()?;
141+
let success = output.status.success();
142142

143143
if !context.check && !success {
144-
return Err(CliError::CommandFailed(format_command(&program, &dir, args)));
144+
let stderr = std::str::from_utf8(&output.stderr).unwrap_or("");
145+
return Err(CliError::CommandFailed(
146+
format_command(&program, &dir, args),
147+
String::from(stderr),
148+
));
145149
}
146150

147151
Ok(success)
@@ -177,7 +181,10 @@ fn rustfmt_test(context: &FmtContext) -> Result<(), CliError> {
177181
{
178182
Err(CliError::RustfmtNotInstalled)
179183
} else {
180-
Err(CliError::CommandFailed(format_command(&program, &dir, args)))
184+
Err(CliError::CommandFailed(
185+
format_command(&program, &dir, args),
186+
std::str::from_utf8(&output.stderr).unwrap_or("").to_string(),
187+
))
181188
}
182189
}
183190

src/tools/clippy/clippy_dev/src/lib.rs

+13
Original file line numberDiff line numberDiff line change
@@ -236,6 +236,10 @@ pub struct FileChange {
236236
/// `path` is the relative path to the file on which you want to perform the replacement.
237237
///
238238
/// See `replace_region_in_text` for documentation of the other options.
239+
///
240+
/// # Panics
241+
///
242+
/// Panics if the path could not read or then written
239243
pub fn replace_region_in_file<F>(
240244
path: &Path,
241245
start: &str,
@@ -283,6 +287,10 @@ where
283287
/// .new_lines;
284288
/// assert_eq!("replace_start\na different\ntext\nreplace_end", result);
285289
/// ```
290+
///
291+
/// # Panics
292+
///
293+
/// Panics if start or end is not valid regex
286294
pub fn replace_region_in_text<F>(text: &str, start: &str, end: &str, replace_start: bool, replacements: F) -> FileChange
287295
where
288296
F: FnOnce() -> Vec<String>,
@@ -329,6 +337,11 @@ where
329337
}
330338

331339
/// Returns the path to the Clippy project directory
340+
///
341+
/// # Panics
342+
///
343+
/// Panics if the current directory could not be retrieved, there was an error reading any of the
344+
/// Cargo.toml files or ancestor directory is the clippy root directory
332345
#[must_use]
333346
pub fn clippy_project_root() -> PathBuf {
334347
let current_dir = std::env::current_dir().unwrap();

src/tools/clippy/clippy_dev/src/ra_setup.rs

+3
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,9 @@ use std::path::{Path, PathBuf};
88
// This allows rust analyzer to analyze rustc internals and show proper information inside clippy
99
// code. See https://github.com/rust-analyzer/rust-analyzer/issues/3517 and https://github.com/rust-lang/rust-clippy/issues/5514 for details
1010

11+
/// # Panics
12+
///
13+
/// Panics if `rustc_path` does not lead to a rustc repo or the files could not be read
1114
pub fn run(rustc_path: Option<&str>) {
1215
// we can unwrap here because the arg is required by clap
1316
let rustc_path = PathBuf::from(rustc_path.unwrap());

src/tools/clippy/clippy_dev/src/serve.rs

+3
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,9 @@ use std::process::Command;
44
use std::thread;
55
use std::time::{Duration, SystemTime};
66

7+
/// # Panics
8+
///
9+
/// Panics if the python commands could not be spawned
710
pub fn run(port: u16, lint: Option<&str>) -> ! {
811
let mut url = Some(match lint {
912
None => format!("http://localhost:{}", port),

src/tools/clippy/clippy_lints/src/doc.rs

+120-5
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,7 @@
1-
use crate::utils::{implements_trait, is_entrypoint_fn, is_type_diagnostic_item, return_ty, span_lint};
1+
use crate::utils::{
2+
implements_trait, is_entrypoint_fn, is_type_diagnostic_item, match_panic_def_id, method_chain_args, return_ty,
3+
span_lint, span_lint_and_note,
4+
};
25
use if_chain::if_chain;
36
use itertools::Itertools;
47
use rustc_ast::ast::{Async, AttrKind, Attribute, FnKind, FnRetTy, ItemKind};
@@ -8,7 +11,10 @@ use rustc_data_structures::sync::Lrc;
811
use rustc_errors::emitter::EmitterWriter;
912
use rustc_errors::Handler;
1013
use rustc_hir as hir;
14+
use rustc_hir::intravisit::{self, NestedVisitorMap, Visitor};
15+
use rustc_hir::{Expr, ExprKind, QPath};
1116
use rustc_lint::{LateContext, LateLintPass};
17+
use rustc_middle::hir::map::Map;
1218
use rustc_middle::lint::in_external_macro;
1319
use rustc_middle::ty;
1420
use rustc_parse::maybe_new_parser_from_source_str;
@@ -122,6 +128,37 @@ declare_clippy_lint! {
122128
"`pub fn` returns `Result` without `# Errors` in doc comment"
123129
}
124130

131+
declare_clippy_lint! {
132+
/// **What it does:** Checks the doc comments of publicly visible functions that
133+
/// may panic and warns if there is no `# Panics` section.
134+
///
135+
/// **Why is this bad?** Documenting the scenarios in which panicking occurs
136+
/// can help callers who do not want to panic to avoid those situations.
137+
///
138+
/// **Known problems:** None.
139+
///
140+
/// **Examples:**
141+
///
142+
/// Since the following function may panic it has a `# Panics` section in
143+
/// its doc comment:
144+
///
145+
/// ```rust
146+
/// /// # Panics
147+
/// ///
148+
/// /// Will panic if y is 0
149+
/// pub fn divide_by(x: i32, y: i32) -> i32 {
150+
/// if y == 0 {
151+
/// panic!("Cannot divide by 0")
152+
/// } else {
153+
/// x / y
154+
/// }
155+
/// }
156+
/// ```
157+
pub MISSING_PANICS_DOC,
158+
pedantic,
159+
"`pub fn` may panic without `# Panics` in doc comment"
160+
}
161+
125162
declare_clippy_lint! {
126163
/// **What it does:** Checks for `fn main() { .. }` in doctests
127164
///
@@ -166,7 +203,9 @@ impl DocMarkdown {
166203
}
167204
}
168205

169-
impl_lint_pass!(DocMarkdown => [DOC_MARKDOWN, MISSING_SAFETY_DOC, MISSING_ERRORS_DOC, NEEDLESS_DOCTEST_MAIN]);
206+
impl_lint_pass!(DocMarkdown =>
207+
[DOC_MARKDOWN, MISSING_SAFETY_DOC, MISSING_ERRORS_DOC, MISSING_PANICS_DOC, NEEDLESS_DOCTEST_MAIN]
208+
);
170209

171210
impl<'tcx> LateLintPass<'tcx> for DocMarkdown {
172211
fn check_crate(&mut self, cx: &LateContext<'tcx>, krate: &'tcx hir::Crate<'_>) {
@@ -180,7 +219,15 @@ impl<'tcx> LateLintPass<'tcx> for DocMarkdown {
180219
if !(is_entrypoint_fn(cx, cx.tcx.hir().local_def_id(item.hir_id).to_def_id())
181220
|| in_external_macro(cx.tcx.sess, item.span))
182221
{
183-
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, Some(body_id));
222+
let body = cx.tcx.hir().body(body_id);
223+
let impl_item_def_id = cx.tcx.hir().local_def_id(item.hir_id);
224+
let mut fpu = FindPanicUnwrap {
225+
cx,
226+
typeck_results: cx.tcx.typeck(impl_item_def_id),
227+
panic_span: None,
228+
};
229+
fpu.visit_expr(&body.value);
230+
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, Some(body_id), fpu.panic_span);
184231
}
185232
},
186233
hir::ItemKind::Impl(ref impl_) => {
@@ -200,7 +247,7 @@ impl<'tcx> LateLintPass<'tcx> for DocMarkdown {
200247
let headers = check_attrs(cx, &self.valid_idents, &item.attrs);
201248
if let hir::TraitItemKind::Fn(ref sig, ..) = item.kind {
202249
if !in_external_macro(cx.tcx.sess, item.span) {
203-
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, None);
250+
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, None, None);
204251
}
205252
}
206253
}
@@ -211,7 +258,15 @@ impl<'tcx> LateLintPass<'tcx> for DocMarkdown {
211258
return;
212259
}
213260
if let hir::ImplItemKind::Fn(ref sig, body_id) = item.kind {
214-
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, Some(body_id));
261+
let body = cx.tcx.hir().body(body_id);
262+
let impl_item_def_id = cx.tcx.hir().local_def_id(item.hir_id);
263+
let mut fpu = FindPanicUnwrap {
264+
cx,
265+
typeck_results: cx.tcx.typeck(impl_item_def_id),
266+
panic_span: None,
267+
};
268+
fpu.visit_expr(&body.value);
269+
lint_for_missing_headers(cx, item.hir_id, item.span, sig, headers, Some(body_id), fpu.panic_span);
215270
}
216271
}
217272
}
@@ -223,6 +278,7 @@ fn lint_for_missing_headers<'tcx>(
223278
sig: &hir::FnSig<'_>,
224279
headers: DocHeaders,
225280
body_id: Option<hir::BodyId>,
281+
panic_span: Option<Span>,
226282
) {
227283
if !cx.access_levels.is_exported(hir_id) {
228284
return; // Private functions do not require doc comments
@@ -235,6 +291,16 @@ fn lint_for_missing_headers<'tcx>(
235291
"unsafe function's docs miss `# Safety` section",
236292
);
237293
}
294+
if !headers.panics && panic_span.is_some() {
295+
span_lint_and_note(
296+
cx,
297+
MISSING_PANICS_DOC,
298+
span,
299+
"docs for function which may panic missing `# Panics` section",
300+
panic_span,
301+
"first possible panic found here",
302+
);
303+
}
238304
if !headers.errors {
239305
if is_type_diagnostic_item(cx, return_ty(cx, hir_id), sym::result_type) {
240306
span_lint(
@@ -321,6 +387,7 @@ pub fn strip_doc_comment_decoration(doc: &str, comment_kind: CommentKind, span:
321387
struct DocHeaders {
322388
safety: bool,
323389
errors: bool,
390+
panics: bool,
324391
}
325392

326393
fn check_attrs<'a>(cx: &LateContext<'_>, valid_idents: &FxHashSet<String>, attrs: &'a [Attribute]) -> DocHeaders {
@@ -338,6 +405,7 @@ fn check_attrs<'a>(cx: &LateContext<'_>, valid_idents: &FxHashSet<String>, attrs
338405
return DocHeaders {
339406
safety: true,
340407
errors: true,
408+
panics: true,
341409
};
342410
}
343411
}
@@ -353,6 +421,7 @@ fn check_attrs<'a>(cx: &LateContext<'_>, valid_idents: &FxHashSet<String>, attrs
353421
return DocHeaders {
354422
safety: false,
355423
errors: false,
424+
panics: false,
356425
};
357426
}
358427

@@ -394,6 +463,7 @@ fn check_doc<'a, Events: Iterator<Item = (pulldown_cmark::Event<'a>, Range<usize
394463
let mut headers = DocHeaders {
395464
safety: false,
396465
errors: false,
466+
panics: false,
397467
};
398468
let mut in_code = false;
399469
let mut in_link = None;
@@ -439,6 +509,7 @@ fn check_doc<'a, Events: Iterator<Item = (pulldown_cmark::Event<'a>, Range<usize
439509
}
440510
headers.safety |= in_heading && text.trim() == "Safety";
441511
headers.errors |= in_heading && text.trim() == "Errors";
512+
headers.panics |= in_heading && text.trim() == "Panics";
442513
let index = match spans.binary_search_by(|c| c.0.cmp(&range.start)) {
443514
Ok(o) => o,
444515
Err(e) => e - 1,
@@ -609,3 +680,47 @@ fn check_word(cx: &LateContext<'_>, word: &str, span: Span) {
609680
);
610681
}
611682
}
683+
684+
struct FindPanicUnwrap<'a, 'tcx> {
685+
cx: &'a LateContext<'tcx>,
686+
panic_span: Option<Span>,
687+
typeck_results: &'tcx ty::TypeckResults<'tcx>,
688+
}
689+
690+
impl<'a, 'tcx> Visitor<'tcx> for FindPanicUnwrap<'a, 'tcx> {
691+
type Map = Map<'tcx>;
692+
693+
fn visit_expr(&mut self, expr: &'tcx Expr<'_>) {
694+
if self.panic_span.is_some() {
695+
return;
696+
}
697+
698+
// check for `begin_panic`
699+
if_chain! {
700+
if let ExprKind::Call(ref func_expr, _) = expr.kind;
701+
if let ExprKind::Path(QPath::Resolved(_, ref path)) = func_expr.kind;
702+
if let Some(path_def_id) = path.res.opt_def_id();
703+
if match_panic_def_id(self.cx, path_def_id);
704+
then {
705+
self.panic_span = Some(expr.span);
706+
}
707+
}
708+
709+
// check for `unwrap`
710+
if let Some(arglists) = method_chain_args(expr, &["unwrap"]) {
711+
let reciever_ty = self.typeck_results.expr_ty(&arglists[0][0]).peel_refs();
712+
if is_type_diagnostic_item(self.cx, reciever_ty, sym::option_type)
713+
|| is_type_diagnostic_item(self.cx, reciever_ty, sym::result_type)
714+
{
715+
self.panic_span = Some(expr.span);
716+
}
717+
}
718+
719+
// and check sub-expressions
720+
intravisit::walk_expr(self, expr);
721+
}
722+
723+
fn nested_visit_map(&mut self) -> NestedVisitorMap<Self::Map> {
724+
NestedVisitorMap::OnlyBodies(self.cx.tcx.hir())
725+
}
726+
}

src/tools/clippy/clippy_lints/src/excessive_bools.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -160,15 +160,17 @@ impl EarlyLintPass for ExcessiveBools {
160160
"consider using a state machine or refactoring bools into two-variant enums",
161161
);
162162
}
163-
}
164-
ItemKind::Impl(box ImplKind { of_trait: None, items, .. })
163+
},
164+
ItemKind::Impl(box ImplKind {
165+
of_trait: None, items, ..
166+
})
165167
| ItemKind::Trait(box TraitKind(.., items)) => {
166168
for item in items {
167169
if let AssocItemKind::Fn(box FnKind(_, fn_sig, _, _)) = &item.kind {
168170
self.check_fn_sig(cx, fn_sig, item.span);
169171
}
170172
}
171-
}
173+
},
172174
ItemKind::Fn(box FnKind(_, fn_sig, _, _)) => self.check_fn_sig(cx, fn_sig, item.span),
173175
_ => (),
174176
}

0 commit comments

Comments
 (0)