mirror of
https://github.com/astral-sh/ruff.git
synced 2025-07-07 21:25:08 +00:00
[ty] Raise invalid-exception-caught
even when exception is not captured (#18202)
Some checks are pending
CI / Determine changes (push) Waiting to run
CI / cargo fmt (push) Waiting to run
CI / cargo clippy (push) Blocked by required conditions
CI / cargo test (linux) (push) Blocked by required conditions
CI / cargo test (linux, release) (push) Blocked by required conditions
CI / cargo test (windows) (push) Blocked by required conditions
CI / cargo test (wasm) (push) Blocked by required conditions
CI / cargo build (release) (push) Waiting to run
CI / cargo build (msrv) (push) Blocked by required conditions
CI / cargo fuzz build (push) Blocked by required conditions
CI / fuzz parser (push) Blocked by required conditions
CI / test scripts (push) Blocked by required conditions
CI / ecosystem (push) Blocked by required conditions
CI / Fuzz for new ty panics (push) Blocked by required conditions
CI / cargo shear (push) Blocked by required conditions
CI / python package (push) Waiting to run
CI / pre-commit (push) Waiting to run
CI / mkdocs (push) Waiting to run
CI / formatter instabilities and black similarity (push) Blocked by required conditions
CI / test ruff-lsp (push) Blocked by required conditions
CI / check playground (push) Blocked by required conditions
CI / benchmarks (push) Blocked by required conditions
[ty Playground] Release / publish (push) Waiting to run
Some checks are pending
CI / Determine changes (push) Waiting to run
CI / cargo fmt (push) Waiting to run
CI / cargo clippy (push) Blocked by required conditions
CI / cargo test (linux) (push) Blocked by required conditions
CI / cargo test (linux, release) (push) Blocked by required conditions
CI / cargo test (windows) (push) Blocked by required conditions
CI / cargo test (wasm) (push) Blocked by required conditions
CI / cargo build (release) (push) Waiting to run
CI / cargo build (msrv) (push) Blocked by required conditions
CI / cargo fuzz build (push) Blocked by required conditions
CI / fuzz parser (push) Blocked by required conditions
CI / test scripts (push) Blocked by required conditions
CI / ecosystem (push) Blocked by required conditions
CI / Fuzz for new ty panics (push) Blocked by required conditions
CI / cargo shear (push) Blocked by required conditions
CI / python package (push) Waiting to run
CI / pre-commit (push) Waiting to run
CI / mkdocs (push) Waiting to run
CI / formatter instabilities and black similarity (push) Blocked by required conditions
CI / test ruff-lsp (push) Blocked by required conditions
CI / check playground (push) Blocked by required conditions
CI / benchmarks (push) Blocked by required conditions
[ty Playground] Release / publish (push) Waiting to run
Co-authored-by: Alex Waygood <alex.waygood@gmail.com> Co-authored-by: Carl Meyer <carl@astral.sh>
This commit is contained in:
parent
a2c87c2bc1
commit
8729cb208f
6 changed files with 128 additions and 23 deletions
|
@ -0,0 +1,14 @@
|
||||||
|
def name_1[name_0: name_0](name_2: name_0):
|
||||||
|
try:
|
||||||
|
pass
|
||||||
|
except name_2:
|
||||||
|
pass
|
||||||
|
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
def name_2[T: Any](x: T):
|
||||||
|
try:
|
||||||
|
pass
|
||||||
|
except x:
|
||||||
|
pass
|
||||||
|
|
|
@ -76,6 +76,44 @@ except Error as err:
|
||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Exception with no captured type
|
||||||
|
|
||||||
|
```py
|
||||||
|
try:
|
||||||
|
{}.get("foo")
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
```
|
||||||
|
|
||||||
|
## Exception which catches typevar
|
||||||
|
|
||||||
|
```toml
|
||||||
|
[environment]
|
||||||
|
python-version = "3.12"
|
||||||
|
```
|
||||||
|
|
||||||
|
```py
|
||||||
|
from typing import Callable
|
||||||
|
|
||||||
|
def silence[T: type[BaseException]](
|
||||||
|
func: Callable[[], None],
|
||||||
|
exception_type: T,
|
||||||
|
):
|
||||||
|
try:
|
||||||
|
func()
|
||||||
|
except exception_type as e:
|
||||||
|
reveal_type(e) # revealed: T'instance
|
||||||
|
|
||||||
|
def silence2[T: (
|
||||||
|
type[ValueError],
|
||||||
|
type[TypeError],
|
||||||
|
)](func: Callable[[], None], exception_type: T,):
|
||||||
|
try:
|
||||||
|
func()
|
||||||
|
except exception_type as e:
|
||||||
|
reveal_type(e) # revealed: T'instance
|
||||||
|
```
|
||||||
|
|
||||||
## Invalid exception handlers
|
## Invalid exception handlers
|
||||||
|
|
||||||
```py
|
```py
|
||||||
|
@ -108,6 +146,12 @@ def foo(
|
||||||
# error: [invalid-exception-caught]
|
# error: [invalid-exception-caught]
|
||||||
except z as g:
|
except z as g:
|
||||||
reveal_type(g) # revealed: Unknown
|
reveal_type(g) # revealed: Unknown
|
||||||
|
|
||||||
|
try:
|
||||||
|
{}.get("foo")
|
||||||
|
# error: [invalid-exception-caught]
|
||||||
|
except int:
|
||||||
|
pass
|
||||||
```
|
```
|
||||||
|
|
||||||
## Object raised is not an exception
|
## Object raised is not an exception
|
||||||
|
|
|
@ -43,9 +43,23 @@ except* (KeyboardInterrupt, AttributeError) as e:
|
||||||
reveal_type(e) # revealed: BaseExceptionGroup[KeyboardInterrupt | AttributeError]
|
reveal_type(e) # revealed: BaseExceptionGroup[KeyboardInterrupt | AttributeError]
|
||||||
```
|
```
|
||||||
|
|
||||||
## Invalid `except*` handlers
|
## `except*` with no captured exception type
|
||||||
|
|
||||||
```py
|
```py
|
||||||
|
try:
|
||||||
|
help()
|
||||||
|
except* TypeError:
|
||||||
|
pass
|
||||||
|
```
|
||||||
|
|
||||||
|
## Invalid `except*` handlers with or without a captured exception type
|
||||||
|
|
||||||
|
```py
|
||||||
|
try:
|
||||||
|
help()
|
||||||
|
except* int: # error: [invalid-exception-caught]
|
||||||
|
pass
|
||||||
|
|
||||||
try:
|
try:
|
||||||
help()
|
help()
|
||||||
except* 3 as e: # error: [invalid-exception-caught]
|
except* 3 as e: # error: [invalid-exception-caught]
|
||||||
|
|
|
@ -4745,6 +4745,32 @@ impl<'db> Type<'db> {
|
||||||
}
|
}
|
||||||
Some(builder.build())
|
Some(builder.build())
|
||||||
}
|
}
|
||||||
|
// If there is no bound or constraints on a typevar `T`, `T: object` implicitly, which
|
||||||
|
// has no instance type. Otherwise, synthesize a typevar with bound or constraints
|
||||||
|
// mapped through `to_instance`.
|
||||||
|
Type::TypeVar(typevar) => {
|
||||||
|
let bound_or_constraints = match typevar.bound_or_constraints(db)? {
|
||||||
|
TypeVarBoundOrConstraints::UpperBound(upper_bound) => {
|
||||||
|
TypeVarBoundOrConstraints::UpperBound(upper_bound.to_instance(db)?)
|
||||||
|
}
|
||||||
|
TypeVarBoundOrConstraints::Constraints(constraints) => {
|
||||||
|
let mut builder = UnionBuilder::new(db);
|
||||||
|
for constraint in constraints.elements(db) {
|
||||||
|
builder = builder.add(constraint.to_instance(db)?);
|
||||||
|
}
|
||||||
|
TypeVarBoundOrConstraints::Constraints(builder.build().into_union()?)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
Some(Type::TypeVar(TypeVarInstance::new(
|
||||||
|
db,
|
||||||
|
Name::new(format!("{}'instance", typevar.name(db))),
|
||||||
|
None,
|
||||||
|
Some(bound_or_constraints),
|
||||||
|
typevar.variance(db),
|
||||||
|
None,
|
||||||
|
typevar.kind(db),
|
||||||
|
)))
|
||||||
|
}
|
||||||
Type::Intersection(_) => Some(todo_type!("Type::Intersection.to_instance")),
|
Type::Intersection(_) => Some(todo_type!("Type::Intersection.to_instance")),
|
||||||
Type::BooleanLiteral(_)
|
Type::BooleanLiteral(_)
|
||||||
| Type::BytesLiteral(_)
|
| Type::BytesLiteral(_)
|
||||||
|
@ -4763,7 +4789,6 @@ impl<'db> Type<'db> {
|
||||||
| Type::IntLiteral(_)
|
| Type::IntLiteral(_)
|
||||||
| Type::StringLiteral(_)
|
| Type::StringLiteral(_)
|
||||||
| Type::Tuple(_)
|
| Type::Tuple(_)
|
||||||
| Type::TypeVar(_)
|
|
||||||
| Type::LiteralString
|
| Type::LiteralString
|
||||||
| Type::BoundSuper(_)
|
| Type::BoundSuper(_)
|
||||||
| Type::AlwaysTruthy
|
| Type::AlwaysTruthy
|
||||||
|
@ -4890,7 +4915,7 @@ impl<'db> Type<'db> {
|
||||||
Ok(Type::TypeVar(TypeVarInstance::new(
|
Ok(Type::TypeVar(TypeVarInstance::new(
|
||||||
db,
|
db,
|
||||||
ast::name::Name::new("Self"),
|
ast::name::Name::new("Self"),
|
||||||
class_def,
|
Some(class_def),
|
||||||
Some(TypeVarBoundOrConstraints::UpperBound(instance)),
|
Some(TypeVarBoundOrConstraints::UpperBound(instance)),
|
||||||
TypeVarVariance::Invariant,
|
TypeVarVariance::Invariant,
|
||||||
None,
|
None,
|
||||||
|
@ -5431,7 +5456,7 @@ impl<'db> Type<'db> {
|
||||||
}
|
}
|
||||||
Self::KnownInstance(instance) => match instance {
|
Self::KnownInstance(instance) => match instance {
|
||||||
KnownInstanceType::TypeVar(var) => {
|
KnownInstanceType::TypeVar(var) => {
|
||||||
Some(TypeDefinition::TypeVar(var.definition(db)))
|
Some(TypeDefinition::TypeVar(var.definition(db)?))
|
||||||
}
|
}
|
||||||
KnownInstanceType::TypeAliasType(type_alias) => {
|
KnownInstanceType::TypeAliasType(type_alias) => {
|
||||||
type_alias.definition(db).map(TypeDefinition::TypeAlias)
|
type_alias.definition(db).map(TypeDefinition::TypeAlias)
|
||||||
|
@ -5457,7 +5482,7 @@ impl<'db> Type<'db> {
|
||||||
| Self::BoundSuper(_)
|
| Self::BoundSuper(_)
|
||||||
| Self::Tuple(_) => self.to_meta_type(db).definition(db),
|
| Self::Tuple(_) => self.to_meta_type(db).definition(db),
|
||||||
|
|
||||||
Self::TypeVar(var) => Some(TypeDefinition::TypeVar(var.definition(db))),
|
Self::TypeVar(var) => Some(TypeDefinition::TypeVar(var.definition(db)?)),
|
||||||
|
|
||||||
Self::ProtocolInstance(protocol) => match protocol.inner {
|
Self::ProtocolInstance(protocol) => match protocol.inner {
|
||||||
Protocol::FromClass(class) => Some(TypeDefinition::Class(class.definition(db))),
|
Protocol::FromClass(class) => Some(TypeDefinition::Class(class.definition(db))),
|
||||||
|
@ -5806,8 +5831,8 @@ pub struct TypeVarInstance<'db> {
|
||||||
#[returns(ref)]
|
#[returns(ref)]
|
||||||
name: ast::name::Name,
|
name: ast::name::Name,
|
||||||
|
|
||||||
/// The type var's definition
|
/// The type var's definition (None if synthesized)
|
||||||
pub definition: Definition<'db>,
|
pub definition: Option<Definition<'db>>,
|
||||||
|
|
||||||
/// The upper bound or constraint on the type of this TypeVar
|
/// The upper bound or constraint on the type of this TypeVar
|
||||||
bound_or_constraints: Option<TypeVarBoundOrConstraints<'db>>,
|
bound_or_constraints: Option<TypeVarBoundOrConstraints<'db>>,
|
||||||
|
|
|
@ -1820,10 +1820,13 @@ impl<'db> BindingError<'db> {
|
||||||
typevar.name(context.db()),
|
typevar.name(context.db()),
|
||||||
));
|
));
|
||||||
|
|
||||||
let typevar_range = typevar.definition(context.db()).full_range(context.db());
|
if let Some(typevar_definition) = typevar.definition(context.db()) {
|
||||||
let mut sub = SubDiagnostic::new(Severity::Info, "Type variable defined here");
|
let typevar_range = typevar_definition.full_range(context.db());
|
||||||
sub.annotate(Annotation::primary(typevar_range.into()));
|
let mut sub = SubDiagnostic::new(Severity::Info, "Type variable defined here");
|
||||||
diag.sub(sub);
|
sub.annotate(Annotation::primary(typevar_range.into()));
|
||||||
|
diag.sub(sub);
|
||||||
|
}
|
||||||
|
|
||||||
if let Some(union_diag) = union_diag {
|
if let Some(union_diag) = union_diag {
|
||||||
union_diag.add_union_context(context.db(), &mut diag);
|
union_diag.add_union_context(context.db(), &mut diag);
|
||||||
}
|
}
|
||||||
|
|
|
@ -2452,7 +2452,7 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
if symbol_name.is_some() {
|
if symbol_name.is_some() {
|
||||||
self.infer_definition(handler);
|
self.infer_definition(handler);
|
||||||
} else {
|
} else {
|
||||||
self.infer_optional_expression(handled_exceptions.as_deref());
|
self.infer_exception(handled_exceptions.as_deref(), try_statement.is_star);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.infer_body(body);
|
self.infer_body(body);
|
||||||
|
@ -2555,11 +2555,7 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn infer_except_handler_definition(
|
fn infer_exception(&mut self, node: Option<&ast::Expr>, is_star: bool) -> Type<'db> {
|
||||||
&mut self,
|
|
||||||
except_handler_definition: &ExceptHandlerDefinitionKind,
|
|
||||||
definition: Definition<'db>,
|
|
||||||
) {
|
|
||||||
fn extract_tuple_specialization<'db>(db: &'db dyn Db, ty: Type<'db>) -> Option<Type<'db>> {
|
fn extract_tuple_specialization<'db>(db: &'db dyn Db, ty: Type<'db>) -> Option<Type<'db>> {
|
||||||
let class = ty.into_nominal_instance()?.class;
|
let class = ty.into_nominal_instance()?.class;
|
||||||
if !class.is_known(db, KnownClass::Tuple) {
|
if !class.is_known(db, KnownClass::Tuple) {
|
||||||
|
@ -2576,8 +2572,6 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
.then_some(specialization_instance)
|
.then_some(specialization_instance)
|
||||||
}
|
}
|
||||||
|
|
||||||
let node = except_handler_definition.handled_exceptions();
|
|
||||||
|
|
||||||
// If there is no handled exception, it's invalid syntax;
|
// If there is no handled exception, it's invalid syntax;
|
||||||
// a diagnostic will have already been emitted
|
// a diagnostic will have already been emitted
|
||||||
let node_ty = node.map_or(Type::unknown(), |ty| self.infer_expression(ty));
|
let node_ty = node.map_or(Type::unknown(), |ty| self.infer_expression(ty));
|
||||||
|
@ -2632,7 +2626,7 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
Type::unknown()
|
Type::unknown()
|
||||||
};
|
};
|
||||||
|
|
||||||
let symbol_ty = if except_handler_definition.is_star() {
|
if is_star {
|
||||||
let class = if symbol_ty
|
let class = if symbol_ty
|
||||||
.is_subtype_of(self.db(), KnownClass::Exception.to_instance(self.db()))
|
.is_subtype_of(self.db(), KnownClass::Exception.to_instance(self.db()))
|
||||||
{
|
{
|
||||||
|
@ -2643,7 +2637,18 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
class.to_specialized_instance(self.db(), [symbol_ty])
|
class.to_specialized_instance(self.db(), [symbol_ty])
|
||||||
} else {
|
} else {
|
||||||
symbol_ty
|
symbol_ty
|
||||||
};
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn infer_except_handler_definition(
|
||||||
|
&mut self,
|
||||||
|
except_handler_definition: &ExceptHandlerDefinitionKind,
|
||||||
|
definition: Definition<'db>,
|
||||||
|
) {
|
||||||
|
let symbol_ty = self.infer_exception(
|
||||||
|
except_handler_definition.handled_exceptions(),
|
||||||
|
except_handler_definition.is_star(),
|
||||||
|
);
|
||||||
|
|
||||||
self.add_binding(
|
self.add_binding(
|
||||||
except_handler_definition.node().into(),
|
except_handler_definition.node().into(),
|
||||||
|
@ -2706,7 +2711,7 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
let ty = Type::KnownInstance(KnownInstanceType::TypeVar(TypeVarInstance::new(
|
let ty = Type::KnownInstance(KnownInstanceType::TypeVar(TypeVarInstance::new(
|
||||||
self.db(),
|
self.db(),
|
||||||
name.id.clone(),
|
name.id.clone(),
|
||||||
definition,
|
Some(definition),
|
||||||
bound_or_constraint,
|
bound_or_constraint,
|
||||||
TypeVarVariance::Invariant, // TODO: infer this
|
TypeVarVariance::Invariant, // TODO: infer this
|
||||||
default_ty,
|
default_ty,
|
||||||
|
@ -5361,7 +5366,7 @@ impl<'db> TypeInferenceBuilder<'db> {
|
||||||
KnownInstanceType::TypeVar(TypeVarInstance::new(
|
KnownInstanceType::TypeVar(TypeVarInstance::new(
|
||||||
self.db(),
|
self.db(),
|
||||||
target.id.clone(),
|
target.id.clone(),
|
||||||
containing_assignment,
|
Some(containing_assignment),
|
||||||
bound_or_constraint,
|
bound_or_constraint,
|
||||||
variance,
|
variance,
|
||||||
*default,
|
*default,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue