Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[red-knot] Implement more types in binary and unary expressions #13803

Merged
merged 23 commits into from
Oct 20, 2024
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
Show all changes
23 commits
Select commit Hold shift + click to select a range
cd88837
Implement unary op for any and unknown
Glyphack Oct 17, 2024
c612089
Handle division by zero for boolean rhs
Glyphack Oct 17, 2024
0fcf286
Handle booleans in binary expression
Glyphack Oct 17, 2024
b056730
Handle instances in unary operation
Glyphack Oct 17, 2024
00df2f9
Remove unknown test case
Glyphack Oct 17, 2024
cad456f
Add more tests without unnecessary variables
Glyphack Oct 18, 2024
3aa7b6c
Check for Bool class in division by zero
Glyphack Oct 19, 2024
de190b8
Convert booleans to integer literal in binary operations
Glyphack Oct 19, 2024
729ddb7
Separate handling unary not for instance types
Glyphack Oct 19, 2024
097678b
Merge branch 'main' into unary-op
Glyphack Oct 19, 2024
ce45e6b
Update crates/red_knot_python_semantic/src/types/infer.rs
Glyphack Oct 19, 2024
f4f425f
Update crates/red_knot_python_semantic/src/types/infer.rs
Glyphack Oct 19, 2024
21e249e
Update crates/red_knot_python_semantic/src/types/infer.rs
Glyphack Oct 19, 2024
05df52f
Apply suggestions
Glyphack Oct 19, 2024
24b8a43
Add tests for integer power op
Glyphack Oct 19, 2024
aca5e41
Merge branch 'main' into unary-op
Glyphack Oct 19, 2024
d962bce
Correct return type
Glyphack Oct 19, 2024
f0b4eef
Update integers.md
Glyphack Oct 19, 2024
d1b3abf
Reformat markdown tests
Glyphack Oct 19, 2024
35555c3
Add debug assert for instance not op
Glyphack Oct 19, 2024
f99644e
Use unreachable
Glyphack Oct 19, 2024
6f8f74b
Update crates/red_knot_python_semantic/src/types/infer.rs
Glyphack Oct 19, 2024
8292567
Apply suggestions from code review
carljm Oct 20, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
## Binary operations on booleans

## Basic Arithmetic

We try to be precise and all operations except for division will result in Literal type.

```py
a = True
b = False

reveal_type(a + a) # revealed: Literal[2]
reveal_type(a + b) # revealed: Literal[1]
reveal_type(b + a) # revealed: Literal[1]
reveal_type(b + b) # revealed: Literal[0]

reveal_type(a - a) # revealed: Literal[0]
reveal_type(a - b) # revealed: Literal[1]
reveal_type(b - a) # revealed: Literal[-1]
reveal_type(b - b) # revealed: Literal[0]

reveal_type(a * a) # revealed: Literal[1]
reveal_type(a * b) # revealed: Literal[0]
reveal_type(b * a) # revealed: Literal[0]
reveal_type(b * b) # revealed: Literal[0]

reveal_type(a % a) # revealed: Literal[0]
reveal_type(b % a) # revealed: Literal[0]

reveal_type(a // a) # revealed: Literal[1]
reveal_type(b // a) # revealed: Literal[0]

reveal_type(a**a) # revealed: Literal[1]
reveal_type(a**b) # revealed: Literal[1]
reveal_type(b**a) # revealed: Literal[0]
reveal_type(b**b) # revealed: Literal[1]

# Division
reveal_type(a / a) # revealed: float
reveal_type(b / a) # revealed: float
b / b # error: [division-by-zero] "Cannot divide object of type `Literal[False]` by zero"
a / b # error: [division-by-zero] "Cannot divide object of type `Literal[True]` by zero"

# bitwise OR
reveal_type(a | a) # revealed: Literal[True]
reveal_type(a | b) # revealed: Literal[True]
reveal_type(b | a) # revealed: Literal[True]
reveal_type(b | b) # revealed: Literal[False]
```
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,18 @@ reveal_type(-3 / 3) # revealed: float
reveal_type(5 % 3) # revealed: Literal[2]
```

## Power

For power if the result fits in the int literal type it will be a Literal type. Otherwise the
outcome is int.

```py
larget_u32 = 4_294_967_295
reveal_type(2**2) # revealed: Literal[4]
reveal_type(1 ** (larget_u32 + 1)) # revealed: int
reveal_type(2**larget_u32) # revealed: int
carljm marked this conversation as resolved.
Show resolved Hide resolved
```

## Division by Zero

This error is really outside the current Python type system, because e.g. `int.__truediv__` and
Expand Down Expand Up @@ -38,6 +50,12 @@ reveal_type(c) # revealed: int
# revealed: float
reveal_type(int() / 0)

# error: "Cannot divide object of type `Literal[1]` by zero"
# revealed: float
reveal_type(1 / False)
True / False # error: [division-by-zero] "Cannot divide object of type `Literal[True]` by zero"
bool(1) / False # error: [division-by-zero] "Cannot divide object of type `Literal[True]` by zero"
carljm marked this conversation as resolved.
Show resolved Hide resolved

# error: "Cannot divide object of type `float` by zero"
# revealed: float
reveal_type(1.0 / 0)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
# Unary Operations

```py
class Number:
Glyphack marked this conversation as resolved.
Show resolved Hide resolved
def __init__(self, value: int):
self.value = 1

def __pos__(self) -> int:
return +self.value

def __neg__(self) -> int:
return -self.value

def __invert__(self) -> Literal[True]:
return True


a = Number()

reveal_type(+a) # revealed: int
reveal_type(-a) # revealed: int
reveal_type(~a) # revealed: @Todo


class NoDunder: ...


b = NoDunder()
+b # error: [unsupported-operator] "Unary operator `+` is unsupported for type `NoDunder`"
-b # error: [unsupported-operator] "Unary operator `-` is unsupported for type `NoDunder`"
~b # error: [unsupported-operator] "Unary operator `~` is unsupported for type `NoDunder`"
```
70 changes: 65 additions & 5 deletions crates/red_knot_python_semantic/src/types/infer.rs
Original file line number Diff line number Diff line change
Expand Up @@ -539,10 +539,11 @@ impl<'db> TypeInferenceBuilder<'db> {
/// Expects the resolved type of the left side of the binary expression.
fn check_division_by_zero(&mut self, expr: &ast::ExprBinOp, left: Type<'db>) {
match left {
Type::IntLiteral(_) => {}
Type::BooleanLiteral(_) | Type::IntLiteral(_) => {}
Type::Instance(cls)
if cls.is_known(self.db, KnownClass::Float)
|| cls.is_known(self.db, KnownClass::Int) => {}
if [KnownClass::Float, KnownClass::Int, KnownClass::Bool]
.iter()
.any(|&k| cls.is_known(self.db, k)) => {}
_ => return,
};

Expand Down Expand Up @@ -2459,7 +2460,9 @@ impl<'db> TypeInferenceBuilder<'db> {
operand,
} = unary;

match (op, self.infer_expression(operand)) {
let operand_type = self.infer_expression(operand);

match (op, operand_type) {
(UnaryOp::UAdd, Type::IntLiteral(value)) => Type::IntLiteral(value),
(UnaryOp::USub, Type::IntLiteral(value)) => Type::IntLiteral(-value),
(UnaryOp::Invert, Type::IntLiteral(value)) => Type::IntLiteral(!value),
Expand All @@ -2469,7 +2472,39 @@ impl<'db> TypeInferenceBuilder<'db> {
(UnaryOp::Invert, Type::BooleanLiteral(bool)) => Type::IntLiteral(!i64::from(bool)),

(UnaryOp::Not, ty) => ty.bool(self.db).negate().into_type(self.db),
(_, Type::Any) => Type::Any,
(_, Type::Unknown) => Type::Unknown,
(op @ (UnaryOp::UAdd | UnaryOp::USub | UnaryOp::Invert), Type::Instance(class)) => {
let unary_dunder_method = match op {
UnaryOp::Invert => "__invert__",
UnaryOp::UAdd => "__pos__",
UnaryOp::USub => "__neg__",
UnaryOp::Not => {
debug_assert!(
AlexWaygood marked this conversation as resolved.
Show resolved Hide resolved
false,
"Not operator should not be handled by dunder method call"
);
return Type::Unknown;
}
};
let class_member = class.class_member(self.db, unary_dunder_method);
let call = class_member.call(self.db, &[operand_type]);
carljm marked this conversation as resolved.
Show resolved Hide resolved

match call.return_ty_result(self.db, AnyNodeRef::ExprUnaryOp(unary), self) {
Ok(t) => t,
Err(e) => {
self.add_diagnostic(
unary.into(),
"unsupported-operator",
format_args!(
"Unary operator `{op}` is unsupported for type `{}`",
operand_type.display(self.db),
),
);
e.return_ty()
}
}
}
_ => Type::Todo, // TODO other unary op types
}
}
Expand All @@ -2491,7 +2526,7 @@ impl<'db> TypeInferenceBuilder<'db> {
(op, right_ty),
(
ast::Operator::Div | ast::Operator::FloorDiv | ast::Operator::Mod,
Type::IntLiteral(0),
Type::IntLiteral(0) | Type::BooleanLiteral(false)
)
) {
self.check_division_by_zero(binary, left_ty);
Expand Down Expand Up @@ -2558,6 +2593,17 @@ impl<'db> TypeInferenceBuilder<'db> {
.unwrap_or_else(|| KnownClass::Int.to_instance(self.db)),
),

(Type::IntLiteral(n), Type::IntLiteral(m), ast::Operator::Pow) => {
carljm marked this conversation as resolved.
Show resolved Hide resolved
let m = u32::try_from(m);
Some(match m {
Ok(m) => n
.checked_pow(m)
.map(Type::IntLiteral)
.unwrap_or_else(|| KnownClass::Int.to_instance(self.db)),
Err(_) => KnownClass::Int.to_instance(self.db),
})
}

(Type::BytesLiteral(lhs), Type::BytesLiteral(rhs), ast::Operator::Add) => {
Some(Type::BytesLiteral(BytesLiteralType::new(
self.db,
Expand Down Expand Up @@ -2693,6 +2739,20 @@ impl<'db> TypeInferenceBuilder<'db> {
})
}

(
Type::BooleanLiteral(b1),
Type::BooleanLiteral(b2),
ruff_python_ast::Operator::BitOr,
) => Some(Type::BooleanLiteral(b1 | b2)),

(Type::BooleanLiteral(bool_value), right, op) => self.infer_binary_expression_type(
Type::IntLiteral(i64::from(bool_value)),
right,
op,
),
(left, Type::BooleanLiteral(bool_value), op) => {
self.infer_binary_expression_type(left, Type::IntLiteral(i64::from(bool_value)), op)
}
_ => Some(Type::Todo), // TODO
}
}
Expand Down
Loading