Compare commits

...

2 Commits

2 changed files with 47 additions and 61 deletions

View File

@@ -14107,19 +14107,21 @@ impl<'db> UnionType<'db> {
self.try_map(db, |element| element.to_instance(db))
}
pub(crate) fn filter(
self,
db: &'db dyn Db,
mut f: impl FnMut(&Type<'db>) -> bool,
) -> Type<'db> {
self.elements(db)
.iter()
.filter(|ty| f(ty))
.fold(UnionBuilder::new(db), |builder, element| {
builder.add(*element)
})
.recursively_defined(self.recursively_defined(db))
.build()
pub(crate) fn filter(self, db: &'db dyn Db, f: impl FnMut(&Type<'db>) -> bool) -> Type<'db> {
let current = self.elements(db);
let new: Vec<Type<'db>> = current.iter().copied().filter(f).collect();
match new.len() {
0 => Type::Never,
1 => new[0],
len if len == current.len() => Type::Union(self),
_ => new
.iter()
.fold(UnionBuilder::new(db), |builder, element| {
builder.add(*element)
})
.recursively_defined(self.recursively_defined(db))
.build(),
}
}
pub(crate) fn map_with_boundness(

View File

@@ -1027,31 +1027,23 @@ impl<'db, 'ast> NarrowingConstraintsBuilder<'db, 'ast> {
&& rhs_ty.is_singleton(self.db)
{
let is_positive_check = is_positive == (ops[0] == ast::CmpOp::Is);
let filtered: Vec<_> = union
.elements(self.db)
.iter()
.filter(|elem| {
elem.as_nominal_instance()
.and_then(|inst| inst.tuple_spec(self.db))
.and_then(|spec| spec.py_index(self.db, index).ok())
.is_none_or(|el_ty| {
if is_positive_check {
// `is X` context: keep tuples where element could be X
!el_ty.is_disjoint_from(self.db, rhs_ty)
} else {
// `is not X` context: keep tuples where element is not always X
!el_ty.is_subtype_of(self.db, rhs_ty)
}
})
})
.copied()
.collect();
if filtered.len() < union.elements(self.db).len() {
let filtered = union.filter(self.db, |elem| {
elem.as_nominal_instance()
.and_then(|inst| inst.tuple_spec(self.db))
.and_then(|spec| spec.py_index(self.db, index).ok())
.is_none_or(|el_ty| {
if is_positive_check {
// `is X` context: keep tuples where element could be X
!el_ty.is_disjoint_from(self.db, rhs_ty)
} else {
// `is not X` context: keep tuples where element is not always X
!el_ty.is_subtype_of(self.db, rhs_ty)
}
})
});
if filtered != Type::Union(union) {
let place = self.expect_place(&subscript_place_expr);
constraints.insert(
place,
NarrowingConstraint::regular(UnionType::from_elements(self.db, filtered)),
);
constraints.insert(place, NarrowingConstraint::typeguard(filtered));
}
}
@@ -1639,33 +1631,25 @@ impl<'db, 'ast> NarrowingConstraintsBuilder<'db, 'ast> {
}
// Filter the union based on whether each tuple element at the index could match the rhs.
let filtered: Vec<_> = union
.elements(self.db)
.iter()
.filter(|elem| {
elem.as_nominal_instance()
.and_then(|inst| inst.tuple_spec(self.db))
.and_then(|spec| spec.py_index(self.db, index).ok())
.is_none_or(|el_ty| {
if constrain_with_equality {
// Keep tuples where element could be equal to rhs.
!el_ty.is_disjoint_from(self.db, rhs_type)
} else {
// Keep tuples where element is not always equal to rhs.
!el_ty.is_subtype_of(self.db, rhs_type)
}
})
})
.copied()
.collect();
let filtered = union.filter(self.db, |elem| {
elem.as_nominal_instance()
.and_then(|inst| inst.tuple_spec(self.db))
.and_then(|spec| spec.py_index(self.db, index).ok())
.is_none_or(|el_ty| {
if constrain_with_equality {
// Keep tuples where element could be equal to rhs.
!el_ty.is_disjoint_from(self.db, rhs_type)
} else {
// Keep tuples where element is not always equal to rhs.
!el_ty.is_subtype_of(self.db, rhs_type)
}
})
});
// Only create a constraint if we actually narrowed something.
if filtered.len() < union.elements(self.db).len() {
if filtered != rhs_type {
let place = self.expect_place(&subscript_place_expr);
Some((
place,
NarrowingConstraint::regular(UnionType::from_elements(self.db, filtered)),
))
Some((place, NarrowingConstraint::typeguard(filtered)))
} else {
None
}