mirror of
				https://github.com/astral-sh/ruff.git
				synced 2025-10-25 09:28:09 +00:00 
			
		
		
		
	Handle raises with implicit alternate branches (#9377)
Closes https://github.com/astral-sh/ruff/issues/9304#issuecomment-1874739740.
This commit is contained in:
		
							parent
							
								
									fd36754beb
								
							
						
					
					
						commit
						dc5094d42a
					
				
					 5 changed files with 435 additions and 68 deletions
				
			
		|  | @ -2,14 +2,16 @@ use ruff_python_ast::{self as ast, ExceptHandler, Stmt}; | |||
| 
 | ||||
| #[derive(Debug, Clone, Copy, PartialEq, Eq)] | ||||
| pub enum Terminal { | ||||
|     /// There is no known terminal (e.g., an implicit return).
 | ||||
|     /// There is no known terminal.
 | ||||
|     None, | ||||
|     /// There is an implicit return (e.g., a path that doesn't return).
 | ||||
|     Implicit, | ||||
|     /// Every path through the function ends with a `raise` statement.
 | ||||
|     Raise, | ||||
|     /// No path through the function ends with a `return` statement.
 | ||||
|     Return, | ||||
|     /// Every path through the function ends with a `return` or `raise` statement.
 | ||||
|     Explicit, | ||||
|     RaiseOrReturn, | ||||
|     /// At least one path through the function ends with a `return` statement.
 | ||||
|     ConditionalReturn, | ||||
| } | ||||
|  | @ -20,6 +22,19 @@ impl Terminal { | |||
|         Self::from_body(&function.body) | ||||
|     } | ||||
| 
 | ||||
|     /// Returns `true` if the [`Terminal`] behavior includes at least one `return` path.
 | ||||
|     pub fn has_any_return(self) -> bool { | ||||
|         matches!( | ||||
|             self, | ||||
|             Self::Return | Self::RaiseOrReturn | Self::ConditionalReturn | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     /// Returns `true` if the [`Terminal`] behavior includes at least one implicit `return` path.
 | ||||
|     pub fn has_implicit_return(self) -> bool { | ||||
|         matches!(self, Self::None | Self::Implicit | Self::ConditionalReturn) | ||||
|     } | ||||
| 
 | ||||
|     /// Returns the [`Terminal`] behavior of the body, if it can be determined.
 | ||||
|     fn from_body(stmts: &[Stmt]) -> Terminal { | ||||
|         let mut terminal = Terminal::None; | ||||
|  | @ -32,10 +47,10 @@ impl Terminal { | |||
|                         continue; | ||||
|                     } | ||||
| 
 | ||||
|                     terminal = terminal.union(Self::from_body(body)); | ||||
|                     terminal = terminal.and_then(Self::from_body(body)); | ||||
| 
 | ||||
|                     if !sometimes_breaks(body) { | ||||
|                         terminal = terminal.union(Self::from_body(orelse)); | ||||
|                         terminal = terminal.and_then(Self::from_body(orelse)); | ||||
|                     } | ||||
|                 } | ||||
|                 Stmt::If(ast::StmtIf { | ||||
|  | @ -43,7 +58,7 @@ impl Terminal { | |||
|                     elif_else_clauses, | ||||
|                     .. | ||||
|                 }) => { | ||||
|                     let branch_terminal = Terminal::combine( | ||||
|                     let branch_terminal = Terminal::branches( | ||||
|                         std::iter::once(Self::from_body(body)).chain( | ||||
|                             elif_else_clauses | ||||
|                                 .iter() | ||||
|  | @ -55,96 +70,176 @@ impl Terminal { | |||
|                     // `else`)...
 | ||||
|                     if elif_else_clauses.iter().any(|clause| clause.test.is_none()) { | ||||
|                         // And all branches return, then the `if` statement returns.
 | ||||
|                         terminal = terminal.union(branch_terminal); | ||||
|                     } else if branch_terminal.has_return() { | ||||
|                         terminal = terminal.and_then(branch_terminal); | ||||
|                     } else if branch_terminal.has_any_return() { | ||||
|                         // Otherwise, if any branch returns, we know this can't be a
 | ||||
|                         // non-returning function.
 | ||||
|                         terminal = terminal.union(Terminal::ConditionalReturn); | ||||
|                         terminal = terminal.and_then(Terminal::ConditionalReturn); | ||||
|                     } | ||||
|                 } | ||||
|                 Stmt::Match(ast::StmtMatch { cases, .. }) => { | ||||
|                     // Note: we assume the `match` is exhaustive.
 | ||||
|                     terminal = terminal.union(Terminal::combine( | ||||
|                     let branch_terminal = terminal.and_then(Terminal::branches( | ||||
|                         cases.iter().map(|case| Self::from_body(&case.body)), | ||||
|                     )); | ||||
| 
 | ||||
|                     // If the `match` is known to be exhaustive (by way of including a wildcard
 | ||||
|                     // pattern)...
 | ||||
|                     if cases.iter().any(is_wildcard) { | ||||
|                         // And all branches return, then the `match` statement returns.
 | ||||
|                         terminal = terminal.and_then(branch_terminal); | ||||
|                     } else { | ||||
|                         // Otherwise, if any branch returns, we know this can't be a
 | ||||
|                         // non-returning function.
 | ||||
|                         if branch_terminal.has_any_return() { | ||||
|                             terminal = terminal.and_then(Terminal::ConditionalReturn); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 Stmt::Try(ast::StmtTry { | ||||
|                     body, | ||||
|                     handlers, | ||||
|                     orelse, | ||||
|                     finalbody, | ||||
|                     .. | ||||
|                 }) => { | ||||
|                     // If the `finally` block returns, the `try` block must also return.
 | ||||
|                     terminal = terminal.union(Self::from_body(finalbody)); | ||||
|                     // If the body returns, then this can't be a non-returning function. We assume
 | ||||
|                     // that _any_ statement in the body could raise an exception, so we don't
 | ||||
|                     // consider the body to be exhaustive. In other words, we assume the exception
 | ||||
|                     // handlers exist for a reason.
 | ||||
|                     let body_terminal = Self::from_body(body); | ||||
|                     if body_terminal.has_any_return() { | ||||
|                         terminal = terminal.and_then(Terminal::ConditionalReturn); | ||||
|                     } | ||||
| 
 | ||||
|                     // If the else block and all the handlers return, the `try` block must also
 | ||||
|                     // return.
 | ||||
|                     let branch_terminal = | ||||
|                         Terminal::combine(std::iter::once(Self::from_body(orelse)).chain( | ||||
|                             handlers.iter().map(|handler| { | ||||
|                                 let ExceptHandler::ExceptHandler(ast::ExceptHandlerExceptHandler { | ||||
|                                     body, | ||||
|                                     .. | ||||
|                                 }) = handler; | ||||
|                                 Self::from_body(body) | ||||
|                             }), | ||||
|                         )); | ||||
|                     // If the `finally` block returns, the `try` block must also return. (Similarly,
 | ||||
|                     // if the `finally` block raises, the `try` block must also raise.)
 | ||||
|                     terminal = terminal.and_then(Self::from_body(finalbody)); | ||||
| 
 | ||||
|                     let branch_terminal = Terminal::branches(handlers.iter().map(|handler| { | ||||
|                         let ExceptHandler::ExceptHandler(ast::ExceptHandlerExceptHandler { | ||||
|                             body, | ||||
|                             .. | ||||
|                         }) = handler; | ||||
|                         Self::from_body(body) | ||||
|                     })); | ||||
| 
 | ||||
|                     if orelse.is_empty() { | ||||
|                         // If there's no `else`, we may fall through.
 | ||||
|                         if branch_terminal.has_return() { | ||||
|                             terminal = terminal.union(Terminal::ConditionalReturn); | ||||
|                         // If there's no `else`, we may fall through, so only mark that this can't
 | ||||
|                         // be a non-returning function if any of the branches return.
 | ||||
|                         if branch_terminal.has_any_return() { | ||||
|                             terminal = terminal.and_then(Terminal::ConditionalReturn); | ||||
|                         } | ||||
|                     } else { | ||||
|                         // If there's an `else`, we may not fall through.
 | ||||
|                         terminal = terminal.union(branch_terminal); | ||||
|                         // If there's an `else`, we won't fall through. If all the handlers and
 | ||||
|                         // the `else` block return,, the `try` block also returns.
 | ||||
|                         terminal = | ||||
|                             terminal.and_then(branch_terminal.branch(Terminal::from_body(orelse))); | ||||
|                     } | ||||
|                 } | ||||
|                 Stmt::With(ast::StmtWith { body, .. }) => { | ||||
|                     terminal = terminal.union(Self::from_body(body)); | ||||
|                     terminal = terminal.and_then(Self::from_body(body)); | ||||
|                 } | ||||
|                 Stmt::Return(_) => { | ||||
|                     terminal = terminal.union(Terminal::Explicit); | ||||
|                     terminal = terminal.and_then(Terminal::RaiseOrReturn); | ||||
|                 } | ||||
|                 Stmt::Raise(_) => { | ||||
|                     terminal = terminal.union(Terminal::Raise); | ||||
|                     terminal = terminal.and_then(Terminal::Raise); | ||||
|                 } | ||||
|                 _ => {} | ||||
|             } | ||||
|         } | ||||
|         terminal | ||||
| 
 | ||||
|         match terminal { | ||||
|             Terminal::None => Terminal::Implicit, | ||||
|             _ => terminal, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     /// Returns `true` if the [`Terminal`] behavior includes at least one `return` path.
 | ||||
|     fn has_return(self) -> bool { | ||||
|         matches!( | ||||
|             self, | ||||
|             Self::Return | Self::Explicit | Self::ConditionalReturn | ||||
|         ) | ||||
|     } | ||||
| 
 | ||||
|     /// Combine two [`Terminal`] operators.
 | ||||
|     fn union(self, other: Self) -> Self { | ||||
|     /// Combine two [`Terminal`] operators, with one appearing after the other.
 | ||||
|     fn and_then(self, other: Self) -> Self { | ||||
|         match (self, other) { | ||||
|             // If one of the operators is `None`, the result is the other operator.
 | ||||
|             (Self::None, other) => other, | ||||
|             (other, Self::None) => other, | ||||
|             (Self::Explicit, _) => Self::Explicit, | ||||
|             (_, Self::Explicit) => Self::Explicit, | ||||
| 
 | ||||
|             // If one of the operators is `Implicit`, the result is the other operator.
 | ||||
|             (Self::Implicit, other) => other, | ||||
|             (other, Self::Implicit) => other, | ||||
| 
 | ||||
|             // If both operators are conditional returns, the result is a conditional return.
 | ||||
|             (Self::ConditionalReturn, Self::ConditionalReturn) => Self::ConditionalReturn, | ||||
|             (Self::Raise, Self::ConditionalReturn) => Self::Explicit, | ||||
|             (Self::ConditionalReturn, Self::Raise) => Self::Explicit, | ||||
| 
 | ||||
|             // If one of the operators is `Raise`, then the function ends with an explicit `raise`
 | ||||
|             // or `return` statement.
 | ||||
|             (Self::Raise, Self::ConditionalReturn) => Self::RaiseOrReturn, | ||||
|             (Self::ConditionalReturn, Self::Raise) => Self::RaiseOrReturn, | ||||
| 
 | ||||
|             // If one of the operators is `Return`, then the function returns.
 | ||||
|             (Self::Return, Self::ConditionalReturn) => Self::Return, | ||||
|             (Self::ConditionalReturn, Self::Return) => Self::Return, | ||||
| 
 | ||||
|             // All paths through the function end with a `raise` statement.
 | ||||
|             (Self::Raise, Self::Raise) => Self::Raise, | ||||
| 
 | ||||
|             // All paths through the function end with a `return` statement.
 | ||||
|             (Self::Return, Self::Return) => Self::Return, | ||||
|             (Self::Raise, Self::Return) => Self::Explicit, | ||||
|             (Self::Return, Self::Raise) => Self::Explicit, | ||||
| 
 | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::Raise, Self::Return) => Self::RaiseOrReturn, | ||||
| 
 | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::Return, Self::Raise) => Self::RaiseOrReturn, | ||||
| 
 | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::RaiseOrReturn, _) => Self::RaiseOrReturn, | ||||
|             (_, Self::RaiseOrReturn) => Self::RaiseOrReturn, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     /// Combine two [`Terminal`] operators from different branches.
 | ||||
|     fn branch(self, other: Self) -> Self { | ||||
|         match (self, other) { | ||||
|             // If one of the operators is `None`, the result is the other operator.
 | ||||
|             (Self::None, other) => other, | ||||
|             (other, Self::None) => other, | ||||
| 
 | ||||
|             // If one of the operators is `Implicit`, the other operator should be downgraded.
 | ||||
|             (Self::Implicit, Self::Implicit) => Self::Implicit, | ||||
|             (Self::Implicit, Self::Raise) => Self::Implicit, | ||||
|             (Self::Raise, Self::Implicit) => Self::Implicit, | ||||
|             (Self::Implicit, Self::Return) => Self::ConditionalReturn, | ||||
|             (Self::Return, Self::Implicit) => Self::ConditionalReturn, | ||||
|             (Self::Implicit, Self::RaiseOrReturn) => Self::ConditionalReturn, | ||||
|             (Self::RaiseOrReturn, Self::Implicit) => Self::ConditionalReturn, | ||||
|             (Self::Implicit, Self::ConditionalReturn) => Self::ConditionalReturn, | ||||
|             (Self::ConditionalReturn, Self::Implicit) => Self::ConditionalReturn, | ||||
| 
 | ||||
|             // If both operators are conditional returns, the result is a conditional return.
 | ||||
|             (Self::ConditionalReturn, Self::ConditionalReturn) => Self::ConditionalReturn, | ||||
| 
 | ||||
|             (Self::Raise, Self::ConditionalReturn) => Self::RaiseOrReturn, | ||||
|             (Self::ConditionalReturn, Self::Raise) => Self::RaiseOrReturn, | ||||
| 
 | ||||
|             (Self::Return, Self::ConditionalReturn) => Self::Return, | ||||
|             (Self::ConditionalReturn, Self::Return) => Self::Return, | ||||
| 
 | ||||
|             // All paths through the function end with a `raise` statement.
 | ||||
|             (Self::Raise, Self::Raise) => Self::Raise, | ||||
|             // All paths through the function end with a `return` statement.
 | ||||
|             (Self::Return, Self::Return) => Self::Return, | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::Raise, Self::Return) => Self::RaiseOrReturn, | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::Return, Self::Raise) => Self::RaiseOrReturn, | ||||
|             // All paths through the function end with a `return` or `raise` statement.
 | ||||
|             (Self::RaiseOrReturn, _) => Self::RaiseOrReturn, | ||||
|             (_, Self::RaiseOrReturn) => Self::RaiseOrReturn, | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     /// Combine a series of [`Terminal`] operators.
 | ||||
|     fn combine(iter: impl Iterator<Item = Terminal>) -> Terminal { | ||||
|         iter.fold(Terminal::None, Self::union) | ||||
|     fn branches(iter: impl Iterator<Item = Terminal>) -> Terminal { | ||||
|         iter.fold(Terminal::None, Terminal::branch) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | @ -153,7 +248,7 @@ fn sometimes_breaks(stmts: &[Stmt]) -> bool { | |||
|     for stmt in stmts { | ||||
|         match stmt { | ||||
|             Stmt::For(ast::StmtFor { body, orelse, .. }) => { | ||||
|                 if Terminal::from_body(body).has_return() { | ||||
|                 if Terminal::from_body(body).has_any_return() { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if sometimes_breaks(orelse) { | ||||
|  | @ -161,7 +256,7 @@ fn sometimes_breaks(stmts: &[Stmt]) -> bool { | |||
|                 } | ||||
|             } | ||||
|             Stmt::While(ast::StmtWhile { body, orelse, .. }) => { | ||||
|                 if Terminal::from_body(body).has_return() { | ||||
|                 if Terminal::from_body(body).has_any_return() { | ||||
|                     return false; | ||||
|                 } | ||||
|                 if sometimes_breaks(orelse) { | ||||
|  | @ -232,3 +327,24 @@ fn always_breaks(stmts: &[Stmt]) -> bool { | |||
|     } | ||||
|     false | ||||
| } | ||||
| 
 | ||||
| /// Returns true if the [`MatchCase`] is a wildcard pattern.
 | ||||
| fn is_wildcard(pattern: &ast::MatchCase) -> bool { | ||||
|     /// Returns true if the [`Pattern`] is a wildcard pattern.
 | ||||
|     fn is_wildcard_pattern(pattern: &ast::Pattern) -> bool { | ||||
|         match pattern { | ||||
|             ast::Pattern::MatchValue(_) | ||||
|             | ast::Pattern::MatchSingleton(_) | ||||
|             | ast::Pattern::MatchSequence(_) | ||||
|             | ast::Pattern::MatchMapping(_) | ||||
|             | ast::Pattern::MatchClass(_) | ||||
|             | ast::Pattern::MatchStar(_) => false, | ||||
|             ast::Pattern::MatchAs(ast::PatternMatchAs { pattern, .. }) => pattern.is_none(), | ||||
|             ast::Pattern::MatchOr(ast::PatternMatchOr { patterns, .. }) => { | ||||
|                 patterns.iter().all(is_wildcard_pattern) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     pattern.guard.is_none() && is_wildcard_pattern(&pattern.pattern) | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Charlie Marsh
						Charlie Marsh