Metadata-Version: 2.4
Name: isos
Version: 0.3.3
Summary: Option and Result types for Python.
Author-email: gsmyridis <georgesmyr@icloud.com>
Requires-Python: >=3.12
Description-Content-Type: text/markdown

# isos

`isos` is a lightweight Python library that brings the Result Pattern to your code.
It introduces two core types — Option and Result — which are intended to **behave** as sum types (also known as tagged unions or discriminated unions) that make handling missing values and errors explicit, safe, and expressive.
They **are** not sum types.

Sum types allow you to represent values that can be one of several variants, where each variant can hold different types of data. This is particularly useful for modeling scenarios where a value can be in different states, like presence/absence (Option) or success/failure (Result).

## Why use the Result pattern?

In Python, a function can return `Optional[T]` — either a value of type `T` or `None`.
But unless you enforce strict type checking, there’s no clear indication that the value may be absent.
This often leads to `AttributeError` or `TypeError` when `None` sneaks through.

Similarly, Python uses exceptions to signal errors, but you don’t always know when a function might raise one.
You either have to read the source or wrap everything in a `try/except`.

By returning a `Result` or `Option` instead, you:

- Make absence and errors visible in the type system.
- Force explicit handling of failure cases.
- Pass, transform, and compose results safely.
- Write code that is more predictable, robust, and self-documenting.

## Examples

### Option

```Python
from isos import Some, Null, Option

def find_user(user_id: int) -> Option[str]:
    users = {1: "Alice", 2: "Bob"}
    return Some(users[user_id]) if user_id in users else Null()

# Handling the result:
user = find_user(1)
if user.is_some():
    print(f"Found user: {user.unwrap()}")
else:
    print("User not found")

# You can also provide a default:
name = find_user(42).unwrap_or("Guest")
print(name)  # -> "Guest"
```

### Result

```Python
from isos import Ok, Err, Error, Result
from typing import final

# Define a custom error
@error("Cannot divide by zero")
class DivisionByZero(Error): ...

def safe_divide(a: float, b: float) -> Result[float]:
    if b == 0:
        return Err(DivisionByZero())
    return Ok(a / b)

result = safe_divide(10, 2)

if result.is_ok():
    print(f"Result is {result.unwrap()}")
else:
    print(f"Error: {result.unwrap_error()}")

# Transforming results
result = safe_divide(10, 0).map(lambda x: x * 2)
# Still Err(DivisionByZero)

# Chaining
result = safe_divide(20, 2).and_then(lambda x: safe_divide(x, 2))
print(result)  # -> Ok(5.0)
```

### Pattern Matching

Python 3.10 introduced pattern matching, which works great with Option and Result:

```Python
from isos import Some, Null, Ok, Err

# Pattern matching with Option
def describe_user(user_opt: Option[str]) -> str:
    match user_opt:
        case Some(name):
            return f"User found: {name}"
        case Null():
            return "No user found"
        case _:
            raise TypeError("user_opt is not an Option")

print(describe_user(Some("Alice")))  # -> "User found: Alice"
print(describe_user(Null()))         # -> "No user found"

# Pattern matching with Result
def handle_division(result: Result[float]) -> str:
    match result:
        case Ok(value):
            return f"Success: {value}"
        case Err(error):
            return f"Failed: {error.MESSAGE}"
        case _:
            raise TypeError("result is not a Result")

print(handle_division(safe_divide(10, 2)))  # -> "Success: 5.0"
print(handle_division(safe_divide(1, 0)))   # -> "Failed: Cannot divide by zero"
```

### Comparison Methods

`isos` does not implement the standard comparison operators (`<`, `<=`, `>`, `>=`) for `Option[T]` and `Result[T]` types
This is because the contained values (`T`) might not always implement these operators, which could lead to runtime errors.

Instead `isos` provides explicit safe comparison methods that return `Result[bool]` based on whether the
contained values are comparable.
If they are not, the comparison returns `Err(NotComparableError())`.

```Python
from isos import Some, Ok, Err, NotComparableError

# Safe comparison - handles incomparable types gracefully
result1 = Some(10).less_than(Some(20))          # Ok(True)
result2 = Some(1+2j).less_than(Some(2+3j))     # Err(NotComparableError())

# You can handle the comparison result safely
match result2:
    case Ok(is_less):
        print(f"Comparison result: {is_less}")
    case Err(error):
        print(f"Cannot compare: {error}")

# Unsafe comparison - faster but may raise exceptions
try:
    Some(10).less_than_unsafe(Some(20))         # True
    Some(1+2j).less_than_unsafe(Some(2+3j))    # TypeError!
except TypeError as e:
    print(f"Comparison failed: {e}")
```

Additionally, `isos` provides explicit "unsafe" comparison methods:

```Python
from isos import Some, Null, Ok, Err

# Option comparisons
assert Some(10).less_than_unsafe(Some(20))  # True
assert Null().less_than_unsafe(Some(10))    # True - Null is always less than Some
assert Some(20).less_than_unsafe(Some(10))  # False

# Result comparisons
result1 = safe_divide(10, 2)  # Ok(5.0)
result2 = safe_divide(20, 2)  # Ok(10.0)
assert result1.less_than_unsafe(result2)    # True
assert not result2.less_than_unsafe(result1)  # False

# Other comparison methods
assert Some(10).less_or_equal_unsafe(Some(10))
assert Some(10).greater_than_unsafe(Some(5))
assert Some(10).greater_or_equal_unsafe(Some(10))
```

These methods are marked as "unsafe" because they require the contained values to implement the `<` operator. If you try to compare values that don't support comparison, you'll get a `TypeError`:

```Python
# This will raise TypeError because complex numbers don't support < operator
Some(1+2j).less_than_unsafe(Some(2+3j))
```
