2023-07-02 14:35:19 +02:00
|
|
|
#![feature(try_blocks)]
|
2020-03-04 17:21:07 +09:00
|
|
|
#![allow(unreachable_code)]
|
2022-07-18 09:39:37 +02:00
|
|
|
#![allow(dead_code)]
|
2020-11-23 13:51:04 +01:00
|
|
|
#![allow(clippy::unnecessary_wraps)]
|
2020-03-04 17:21:07 +09:00
|
|
|
|
|
|
|
fn some_func(a: Option<u32>) -> Option<u32> {
|
|
|
|
a?;
|
|
|
|
|
|
|
|
a
|
|
|
|
}
|
|
|
|
|
|
|
|
fn some_other_func(a: Option<u32>) -> Option<u32> {
|
|
|
|
if a.is_none() {
|
|
|
|
return None;
|
|
|
|
} else {
|
|
|
|
return Some(0);
|
|
|
|
}
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub enum SeemsOption<T> {
|
|
|
|
Some(T),
|
|
|
|
None,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> SeemsOption<T> {
|
|
|
|
pub fn is_none(&self) -> bool {
|
|
|
|
match *self {
|
|
|
|
SeemsOption::None => true,
|
|
|
|
SeemsOption::Some(_) => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn returns_something_similar_to_option(a: SeemsOption<u32>) -> SeemsOption<u32> {
|
|
|
|
if a.is_none() {
|
|
|
|
return SeemsOption::None;
|
|
|
|
}
|
|
|
|
|
|
|
|
a
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct CopyStruct {
|
|
|
|
pub opt: Option<u32>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CopyStruct {
|
|
|
|
#[rustfmt::skip]
|
|
|
|
pub fn func(&self) -> Option<u32> {
|
|
|
|
(self.opt)?;
|
|
|
|
|
|
|
|
self.opt?;
|
|
|
|
|
|
|
|
let _ = Some(self.opt?);
|
|
|
|
|
|
|
|
let _ = self.opt?;
|
|
|
|
|
|
|
|
self.opt
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Clone)]
|
|
|
|
pub struct MoveStruct {
|
|
|
|
pub opt: Option<Vec<u32>>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl MoveStruct {
|
|
|
|
pub fn ref_func(&self) -> Option<Vec<u32>> {
|
|
|
|
self.opt.as_ref()?;
|
|
|
|
|
|
|
|
self.opt.clone()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn mov_func_reuse(self) -> Option<Vec<u32>> {
|
|
|
|
self.opt.as_ref()?;
|
|
|
|
|
|
|
|
self.opt
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn mov_func_no_use(self) -> Option<Vec<u32>> {
|
|
|
|
self.opt.as_ref()?;
|
|
|
|
Some(Vec::new())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn if_let_ref_func(self) -> Option<Vec<u32>> {
|
|
|
|
let v: &Vec<_> = self.opt.as_ref()?;
|
|
|
|
|
|
|
|
Some(v.clone())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn if_let_mov_func(self) -> Option<Vec<u32>> {
|
|
|
|
let v = self.opt?;
|
|
|
|
|
|
|
|
Some(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-17 13:09:02 +09:00
|
|
|
fn func() -> Option<i32> {
|
|
|
|
fn f() -> Option<String> {
|
|
|
|
Some(String::new())
|
|
|
|
}
|
|
|
|
|
|
|
|
f()?;
|
|
|
|
|
|
|
|
Some(0)
|
|
|
|
}
|
|
|
|
|
2021-11-04 12:52:36 +00:00
|
|
|
fn func_returning_result() -> Result<i32, i32> {
|
|
|
|
Ok(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn result_func(x: Result<i32, i32>) -> Result<i32, i32> {
|
2021-10-21 13:11:36 +02:00
|
|
|
let _ = x?;
|
|
|
|
|
|
|
|
x?;
|
|
|
|
|
|
|
|
// No warning
|
|
|
|
let y = if let Ok(x) = x {
|
|
|
|
x
|
|
|
|
} else {
|
2021-11-04 12:52:36 +00:00
|
|
|
return Err(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
// issue #7859
|
|
|
|
// no warning
|
|
|
|
let _ = if let Ok(x) = func_returning_result() {
|
|
|
|
x
|
|
|
|
} else {
|
|
|
|
return Err(0);
|
2021-10-21 13:11:36 +02:00
|
|
|
};
|
|
|
|
|
2021-11-04 12:52:36 +00:00
|
|
|
// no warning
|
|
|
|
if func_returning_result().is_err() {
|
|
|
|
return func_returning_result();
|
|
|
|
}
|
|
|
|
|
2022-11-21 20:34:47 +01:00
|
|
|
// no warning
|
|
|
|
let _ = if let Err(e) = x { Err(e) } else { Ok(0) };
|
|
|
|
|
2023-08-11 14:05:13 +02:00
|
|
|
// issue #11283
|
|
|
|
// no warning
|
|
|
|
#[warn(clippy::question_mark_used)]
|
|
|
|
{
|
|
|
|
if let Err(err) = Ok(()) {
|
|
|
|
return Err(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
if Err::<i32, _>(0).is_err() {
|
|
|
|
return Err(0);
|
|
|
|
} else {
|
|
|
|
return Ok(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
unreachable!()
|
|
|
|
}
|
|
|
|
|
2021-10-21 13:11:36 +02:00
|
|
|
Ok(y)
|
|
|
|
}
|
|
|
|
|
2021-12-17 13:40:22 +01:00
|
|
|
// see issue #8019
|
|
|
|
pub enum NotOption {
|
|
|
|
None,
|
|
|
|
First,
|
|
|
|
AfterFirst,
|
|
|
|
}
|
|
|
|
|
|
|
|
fn obj(_: i32) -> Result<(), NotOption> {
|
|
|
|
Err(NotOption::First)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn f() -> NotOption {
|
|
|
|
if obj(2).is_err() {
|
|
|
|
return NotOption::None;
|
|
|
|
}
|
|
|
|
NotOption::First
|
|
|
|
}
|
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
fn do_something() {}
|
2020-03-04 17:21:07 +09:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
fn err_immediate_return() -> Result<i32, i32> {
|
|
|
|
func_returning_result()?;
|
|
|
|
Ok(1)
|
|
|
|
}
|
2020-03-04 17:21:07 +09:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
fn err_immediate_return_and_do_something() -> Result<i32, i32> {
|
|
|
|
func_returning_result()?;
|
|
|
|
do_something();
|
|
|
|
Ok(1)
|
|
|
|
}
|
2020-03-04 17:21:07 +09:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
// No warning
|
|
|
|
fn no_immediate_return() -> Result<i32, i32> {
|
|
|
|
if let Err(err) = func_returning_result() {
|
|
|
|
do_something();
|
|
|
|
return Err(err);
|
|
|
|
}
|
|
|
|
Ok(1)
|
|
|
|
}
|
2020-04-17 13:09:02 +09:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
// No warning
|
|
|
|
fn mixed_result_and_option() -> Option<i32> {
|
|
|
|
if let Err(err) = func_returning_result() {
|
|
|
|
return Some(err);
|
|
|
|
}
|
|
|
|
None
|
|
|
|
}
|
|
|
|
|
|
|
|
// No warning
|
|
|
|
fn else_if_check() -> Result<i32, i32> {
|
|
|
|
if true {
|
|
|
|
Ok(1)
|
|
|
|
} else if let Err(e) = func_returning_result() {
|
|
|
|
Err(e)
|
|
|
|
} else {
|
|
|
|
Err(-1)
|
|
|
|
}
|
|
|
|
}
|
2021-10-21 13:11:36 +02:00
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
// No warning
|
|
|
|
#[allow(clippy::manual_map)]
|
|
|
|
#[rustfmt::skip]
|
|
|
|
fn option_map() -> Option<bool> {
|
|
|
|
if let Some(a) = Some(false) {
|
|
|
|
Some(!a)
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2020-03-04 17:21:07 +09:00
|
|
|
}
|
2022-07-18 09:39:37 +02:00
|
|
|
|
2022-08-31 09:24:45 -04:00
|
|
|
pub struct PatternedError {
|
|
|
|
flag: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
// No warning
|
|
|
|
fn pattern() -> Result<(), PatternedError> {
|
|
|
|
let res = Ok(());
|
|
|
|
|
|
|
|
if let Err(err @ PatternedError { flag: true }) = res {
|
|
|
|
return Err(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
res
|
|
|
|
}
|
|
|
|
|
2022-07-18 09:39:37 +02:00
|
|
|
fn main() {}
|
2022-10-06 09:44:38 +02:00
|
|
|
|
2023-07-02 14:35:19 +02:00
|
|
|
// `?` is not the same as `return None;` if inside of a try block
|
|
|
|
fn issue8628(a: Option<u32>) -> Option<u32> {
|
|
|
|
let b: Option<u32> = try {
|
|
|
|
if a.is_none() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
32
|
|
|
|
};
|
|
|
|
b.or(Some(128))
|
|
|
|
}
|
|
|
|
|
|
|
|
fn issue6828_nested_body() -> Option<u32> {
|
|
|
|
try {
|
|
|
|
fn f2(a: Option<i32>) -> Option<i32> {
|
|
|
|
a?;
|
|
|
|
Some(32)
|
|
|
|
}
|
|
|
|
123
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-06 09:44:38 +02:00
|
|
|
// should not lint, `?` operator not available in const context
|
|
|
|
const fn issue9175(option: Option<()>) -> Option<()> {
|
|
|
|
if option.is_none() {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
//stuff
|
|
|
|
Some(())
|
|
|
|
}
|