mirror of
https://github.com/rust-lang/rust-clippy
synced 2024-11-23 13:13:34 +00:00
new regex syntax lint, fixes #597
This commit is contained in:
parent
b90288eced
commit
3d85cc24e7
7 changed files with 81 additions and 4 deletions
|
@ -19,6 +19,7 @@ plugin = true
|
||||||
[dependencies]
|
[dependencies]
|
||||||
unicode-normalization = "0.1"
|
unicode-normalization = "0.1"
|
||||||
semver = "0.2.1"
|
semver = "0.2.1"
|
||||||
|
regex-syntax = "0.2.2"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
compiletest_rs = "0.0.11"
|
compiletest_rs = "0.0.11"
|
||||||
|
|
|
@ -6,7 +6,7 @@ A collection of lints to catch common mistakes and improve your Rust code.
|
||||||
[Jump to usage instructions](#usage)
|
[Jump to usage instructions](#usage)
|
||||||
|
|
||||||
##Lints
|
##Lints
|
||||||
There are 109 lints included in this crate:
|
There are 111 lints included in this crate:
|
||||||
|
|
||||||
name | default | meaning
|
name | default | meaning
|
||||||
---------------------------------------------------------------------------------------------------------------|---------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
---------------------------------------------------------------------------------------------------------------|---------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|
||||||
|
@ -46,6 +46,7 @@ name
|
||||||
[identity_op](https://github.com/Manishearth/rust-clippy/wiki#identity_op) | warn | using identity operations, e.g. `x + 0` or `y / 1`
|
[identity_op](https://github.com/Manishearth/rust-clippy/wiki#identity_op) | warn | using identity operations, e.g. `x + 0` or `y / 1`
|
||||||
[ineffective_bit_mask](https://github.com/Manishearth/rust-clippy/wiki#ineffective_bit_mask) | warn | expressions where a bit mask will be rendered useless by a comparison, e.g. `(x | 1) > 2`
|
[ineffective_bit_mask](https://github.com/Manishearth/rust-clippy/wiki#ineffective_bit_mask) | warn | expressions where a bit mask will be rendered useless by a comparison, e.g. `(x | 1) > 2`
|
||||||
[inline_always](https://github.com/Manishearth/rust-clippy/wiki#inline_always) | warn | `#[inline(always)]` is a bad idea in most cases
|
[inline_always](https://github.com/Manishearth/rust-clippy/wiki#inline_always) | warn | `#[inline(always)]` is a bad idea in most cases
|
||||||
|
[invalid_regex](https://github.com/Manishearth/rust-clippy/wiki#invalid_regex) | deny | finds invalid regular expressions in `Regex::new(_)` invocations
|
||||||
[items_after_statements](https://github.com/Manishearth/rust-clippy/wiki#items_after_statements) | warn | finds blocks where an item comes after a statement
|
[items_after_statements](https://github.com/Manishearth/rust-clippy/wiki#items_after_statements) | warn | finds blocks where an item comes after a statement
|
||||||
[iter_next_loop](https://github.com/Manishearth/rust-clippy/wiki#iter_next_loop) | warn | for-looping over `_.next()` which is probably not intended
|
[iter_next_loop](https://github.com/Manishearth/rust-clippy/wiki#iter_next_loop) | warn | for-looping over `_.next()` which is probably not intended
|
||||||
[len_without_is_empty](https://github.com/Manishearth/rust-clippy/wiki#len_without_is_empty) | warn | traits and impls that have `.len()` but not `.is_empty()`
|
[len_without_is_empty](https://github.com/Manishearth/rust-clippy/wiki#len_without_is_empty) | warn | traits and impls that have `.len()` but not `.is_empty()`
|
||||||
|
@ -85,6 +86,7 @@ name
|
||||||
[range_zip_with_len](https://github.com/Manishearth/rust-clippy/wiki#range_zip_with_len) | warn | zipping iterator with a range when enumerate() would do
|
[range_zip_with_len](https://github.com/Manishearth/rust-clippy/wiki#range_zip_with_len) | warn | zipping iterator with a range when enumerate() would do
|
||||||
[redundant_closure](https://github.com/Manishearth/rust-clippy/wiki#redundant_closure) | warn | using redundant closures, i.e. `|a| foo(a)` (which can be written as just `foo`)
|
[redundant_closure](https://github.com/Manishearth/rust-clippy/wiki#redundant_closure) | warn | using redundant closures, i.e. `|a| foo(a)` (which can be written as just `foo`)
|
||||||
[redundant_pattern](https://github.com/Manishearth/rust-clippy/wiki#redundant_pattern) | warn | using `name @ _` in a pattern
|
[redundant_pattern](https://github.com/Manishearth/rust-clippy/wiki#redundant_pattern) | warn | using `name @ _` in a pattern
|
||||||
|
[regex_macro](https://github.com/Manishearth/rust-clippy/wiki#regex_macro) | allow | finds use of `regex!(_)`, suggests `Regex::new(_)` instead
|
||||||
[result_unwrap_used](https://github.com/Manishearth/rust-clippy/wiki#result_unwrap_used) | allow | using `Result.unwrap()`, which might be better handled
|
[result_unwrap_used](https://github.com/Manishearth/rust-clippy/wiki#result_unwrap_used) | allow | using `Result.unwrap()`, which might be better handled
|
||||||
[reverse_range_loop](https://github.com/Manishearth/rust-clippy/wiki#reverse_range_loop) | warn | Iterating over an empty range, such as `10..0` or `5..5`
|
[reverse_range_loop](https://github.com/Manishearth/rust-clippy/wiki#reverse_range_loop) | warn | Iterating over an empty range, such as `10..0` or `5..5`
|
||||||
[search_is_some](https://github.com/Manishearth/rust-clippy/wiki#search_is_some) | warn | using an iterator search followed by `is_some()`, which is more succinctly expressed as a call to `any()`
|
[search_is_some](https://github.com/Manishearth/rust-clippy/wiki#search_is_some) | warn | using an iterator search followed by `is_some()`, which is more succinctly expressed as a call to `any()`
|
||||||
|
|
|
@ -28,6 +28,9 @@ extern crate unicode_normalization;
|
||||||
// for semver check in attrs.rs
|
// for semver check in attrs.rs
|
||||||
extern crate semver;
|
extern crate semver;
|
||||||
|
|
||||||
|
// for regex checking
|
||||||
|
extern crate regex_syntax;
|
||||||
|
|
||||||
extern crate rustc_plugin;
|
extern crate rustc_plugin;
|
||||||
|
|
||||||
use rustc_plugin::Registry;
|
use rustc_plugin::Registry;
|
||||||
|
@ -82,6 +85,7 @@ pub mod derive;
|
||||||
pub mod print;
|
pub mod print;
|
||||||
pub mod vec;
|
pub mod vec;
|
||||||
pub mod drop_ref;
|
pub mod drop_ref;
|
||||||
|
pub mod regex;
|
||||||
|
|
||||||
mod reexport {
|
mod reexport {
|
||||||
pub use syntax::ast::{Name, NodeId};
|
pub use syntax::ast::{Name, NodeId};
|
||||||
|
@ -150,7 +154,7 @@ pub fn plugin_registrar(reg: &mut Registry) {
|
||||||
reg.register_late_lint_pass(box vec::UselessVec);
|
reg.register_late_lint_pass(box vec::UselessVec);
|
||||||
reg.register_late_lint_pass(box drop_ref::DropRefPass);
|
reg.register_late_lint_pass(box drop_ref::DropRefPass);
|
||||||
reg.register_late_lint_pass(box types::AbsurdUnsignedComparisons);
|
reg.register_late_lint_pass(box types::AbsurdUnsignedComparisons);
|
||||||
|
reg.register_late_lint_pass(box regex::RegexPass);
|
||||||
|
|
||||||
reg.register_lint_group("clippy_pedantic", vec![
|
reg.register_lint_group("clippy_pedantic", vec![
|
||||||
matches::SINGLE_MATCH_ELSE,
|
matches::SINGLE_MATCH_ELSE,
|
||||||
|
@ -163,7 +167,6 @@ pub fn plugin_registrar(reg: &mut Registry) {
|
||||||
shadow::SHADOW_REUSE,
|
shadow::SHADOW_REUSE,
|
||||||
shadow::SHADOW_SAME,
|
shadow::SHADOW_SAME,
|
||||||
shadow::SHADOW_UNRELATED,
|
shadow::SHADOW_UNRELATED,
|
||||||
strings::STRING_ADD,
|
|
||||||
strings::STRING_ADD_ASSIGN,
|
strings::STRING_ADD_ASSIGN,
|
||||||
types::CAST_POSSIBLE_TRUNCATION,
|
types::CAST_POSSIBLE_TRUNCATION,
|
||||||
types::CAST_POSSIBLE_WRAP,
|
types::CAST_POSSIBLE_WRAP,
|
||||||
|
@ -250,6 +253,7 @@ pub fn plugin_registrar(reg: &mut Registry) {
|
||||||
ptr_arg::PTR_ARG,
|
ptr_arg::PTR_ARG,
|
||||||
ranges::RANGE_STEP_BY_ZERO,
|
ranges::RANGE_STEP_BY_ZERO,
|
||||||
ranges::RANGE_ZIP_WITH_LEN,
|
ranges::RANGE_ZIP_WITH_LEN,
|
||||||
|
regex::INVALID_REGEX,
|
||||||
returns::LET_AND_RETURN,
|
returns::LET_AND_RETURN,
|
||||||
returns::NEEDLESS_RETURN,
|
returns::NEEDLESS_RETURN,
|
||||||
strings::STRING_LIT_AS_BYTES,
|
strings::STRING_LIT_AS_BYTES,
|
||||||
|
|
53
src/regex.rs
Normal file
53
src/regex.rs
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
use regex_syntax;
|
||||||
|
use std::error::Error;
|
||||||
|
use syntax::codemap::{Span, BytePos, Pos};
|
||||||
|
use rustc_front::hir::*;
|
||||||
|
use rustc::middle::const_eval::{eval_const_expr_partial, ConstVal};
|
||||||
|
use rustc::middle::const_eval::EvalHint::ExprTypeChecked;
|
||||||
|
use rustc::lint::*;
|
||||||
|
|
||||||
|
use utils::{match_path, REGEX_NEW_PATH, span_lint};
|
||||||
|
|
||||||
|
/// **What it does:** This lint checks `Regex::new(_)` invocations for correct regex syntax. It is `deny` by default.
|
||||||
|
///
|
||||||
|
/// **Why is this bad?** This will lead to a runtime panic.
|
||||||
|
///
|
||||||
|
/// **Known problems:** None.
|
||||||
|
///
|
||||||
|
/// **Example:** `Regex::new("|")`
|
||||||
|
declare_lint! {
|
||||||
|
pub INVALID_REGEX,
|
||||||
|
Deny,
|
||||||
|
"finds invalid regular expressions in `Regex::new(_)` invocations"
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Copy,Clone)]
|
||||||
|
pub struct RegexPass;
|
||||||
|
|
||||||
|
impl LintPass for RegexPass {
|
||||||
|
fn get_lints(&self) -> LintArray {
|
||||||
|
lint_array!(INVALID_REGEX)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LateLintPass for RegexPass {
|
||||||
|
fn check_expr(&mut self, cx: &LateContext, expr: &Expr) {
|
||||||
|
if_let_chain!{[
|
||||||
|
let ExprCall(ref fun, ref args) = expr.node,
|
||||||
|
let ExprPath(_, ref path) = fun.node,
|
||||||
|
match_path(path, ®EX_NEW_PATH) && args.len() == 1,
|
||||||
|
let Ok(ConstVal::Str(r)) = eval_const_expr_partial(cx.tcx,
|
||||||
|
&*args[0],
|
||||||
|
ExprTypeChecked,
|
||||||
|
None),
|
||||||
|
let Err(e) = regex_syntax::Expr::parse(&r)
|
||||||
|
], {
|
||||||
|
let lo = args[0].span.lo + BytePos::from_usize(e.position());
|
||||||
|
let span = Span{ lo: lo, hi: lo, expn_id: args[0].span.expn_id };
|
||||||
|
span_lint(cx,
|
||||||
|
INVALID_REGEX,
|
||||||
|
span,
|
||||||
|
&format!("Regex syntax error: {}", e.description()));
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
}
|
|
@ -36,6 +36,7 @@ pub const LL_PATH: [&'static str; 3] = ["collections", "linked_list", "LinkedLis
|
||||||
pub const MUTEX_PATH: [&'static str; 4] = ["std", "sync", "mutex", "Mutex"];
|
pub const MUTEX_PATH: [&'static str; 4] = ["std", "sync", "mutex", "Mutex"];
|
||||||
pub const OPEN_OPTIONS_PATH: [&'static str; 3] = ["std", "fs", "OpenOptions"];
|
pub const OPEN_OPTIONS_PATH: [&'static str; 3] = ["std", "fs", "OpenOptions"];
|
||||||
pub const OPTION_PATH: [&'static str; 3] = ["core", "option", "Option"];
|
pub const OPTION_PATH: [&'static str; 3] = ["core", "option", "Option"];
|
||||||
|
pub const REGEX_NEW_PATH: [&'static str; 3] = ["regex", "Regex", "new"];
|
||||||
pub const RESULT_PATH: [&'static str; 3] = ["core", "result", "Result"];
|
pub const RESULT_PATH: [&'static str; 3] = ["core", "result", "Result"];
|
||||||
pub const STRING_PATH: [&'static str; 3] = ["collections", "string", "String"];
|
pub const STRING_PATH: [&'static str; 3] = ["collections", "string", "String"];
|
||||||
pub const VEC_FROM_ELEM_PATH: [&'static str; 3] = ["std", "vec", "from_elem"];
|
pub const VEC_FROM_ELEM_PATH: [&'static str; 3] = ["std", "vec", "from_elem"];
|
||||||
|
|
16
tests/compile-fail/regex.rs
Normal file
16
tests/compile-fail/regex.rs
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
#![feature(plugin)]
|
||||||
|
#![plugin(clippy)]
|
||||||
|
|
||||||
|
#![allow(unused)]
|
||||||
|
#![deny(invalid_regex)]
|
||||||
|
|
||||||
|
extern crate regex;
|
||||||
|
|
||||||
|
use regex::Regex;
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let pipe_in_wrong_position = Regex::new("|");
|
||||||
|
//~^ERROR: Regex syntax error: empty alternate
|
||||||
|
let wrong_char_range = Regex::new("[z-a]");
|
||||||
|
//~^ERROR: Regex syntax error: invalid character class range
|
||||||
|
}
|
|
@ -7,7 +7,7 @@ fn run_mode(mode: &'static str) {
|
||||||
let mut config = compiletest::default_config();
|
let mut config = compiletest::default_config();
|
||||||
|
|
||||||
let cfg_mode = mode.parse().ok().expect("Invalid mode");
|
let cfg_mode = mode.parse().ok().expect("Invalid mode");
|
||||||
config.target_rustcflags = Some("-L target/debug/".to_owned());
|
config.target_rustcflags = Some("-L target/debug/ -L target/debug/deps".to_owned());
|
||||||
if let Ok(name) = var::<&str>("TESTNAME") {
|
if let Ok(name) = var::<&str>("TESTNAME") {
|
||||||
let s : String = name.to_owned();
|
let s : String = name.to_owned();
|
||||||
config.filter = Some(s)
|
config.filter = Some(s)
|
||||||
|
|
Loading…
Reference in a new issue