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

Bind method doesn't propagate error types correctly in chained operations #233

Open
edodo1337 opened this issue Dec 1, 2024 · 1 comment

Comments

@edodo1337
Copy link

edodo1337 commented Dec 1, 2024

The current implementation of the bind method in the Result class does not properly propagate or merge error types when chaining operations. For example:

@dataclass
class Order:
    amount: float
    product: str

def verify_amount(order: Order) -> Result[Order, ValueError]:
    if order.amount < 0:
        return Error(ValueError("Amount cannot be negative"))
    return Ok(order)

def verify_product(order: Order) -> Result[Order, ValueError]:
    if order.product not in ["Apple", "Banana"]:
        return Error(ValueError("Product not found"))
    return Ok(order)

def order_amount_with_commission(order: Order) -> Result[float, ValueError | KeyError]:
    return Ok(order.amount * 1.1)

order = Order(amount=-100, product="Apple1")

result = (
    Ok(order)
    .bind(verify_amount)
    .bind(verify_product)
    .bind(order_amount_with_commission)
)

The expected type of result should be:
Result[float, ValueError | KeyError]

However, the actual inferred type is:
Result[float, Any]

I found that a small change in the bind method's type signature partially solves the issue by propagating and merging error types more accurately. However, Any still remains in the inferred Result type:

def bind(self, mapper: Callable[[_TSourceOut], Result[_TResult, _TError | _TOther]]) -> Result[_TResult, _TError | _TOther]:

Now type inferred as Result[float, ValueError | KeyError | Any]

@dbrattli
Copy link
Owner

Hey,

The way to do this is to give the Ok a proper type either:

ok: Result[Order, ValueError] = Ok(order)

result = (
    ok
    .bind(verify_amount)
    .bind(verify_product)
    .bind(order_amount_with_commission)
)

or:

result = (
    Result[Order, ValueError].Ok(order)
    .bind(verify_amount)
    .bind(verify_product)
    .bind(order_amount_with_commission)
)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

2 participants