Option and Result container for python
documentation: https://jerkos.github.io/fateful/
Python has some great functional features. The most notable ones are list / dict comprehensions. However, when you start to chain function calls (or predicate or whatever), it becomes rapidly a pain.
In some famous languages, we have Option and Result monad helping user to handle optional values (i.e. None values) and possibly failing computation (Rust, Java, Kotlin...)
So, this very small Python library implements those monads - Option, Result or AsyncResult - providing a way to handle optional values / computation failures in a functional style.
A focus has been made to typing (but can still be improved of course !)
pip install fateful
pip install fateful[http] # install aiohttp to enable async api calls helper
pip install fateful[orjson] # install orjson a fast json implementation
pip install fateful[all] # install all optional dependencies
The Option Monad, also known as the Maybe Monad, is a concept from functional programming that provides a way to handle optional values in a more controlled and expressive manner. It helps avoid null or undefined errors by encapsulating a value that may or may not be present. The key idea behind the Option Monad is to provide a container that can either hold a value (Some) or indicate the absence of a value (None).
The Option Monad offers several benefits, including:
- Safety: It helps prevent null or undefined errors that can occur when working with optional values.
- Clarity: It makes the presence or absence of a value explicit, enhancing the readability of code.
- Composability: It allows for chaining operations on optional values without explicitly checking for null or undefined values.
- Functional style: It promotes functional programming principles by encouraging pure functions and immutability.
When you do not know if a value is None or is actually a real value / object, use opt to wrap it into an option
from fateful import opt, Null, Some
# Null is a singleton which is basically Empty(None)
>>> opt('value').or_('new value')
value
>>> opt('value').or_else(lambda: 'another value)
value
>>> Null.or_else(lambda: 'new value')
new value
>>> opt(None).get() # same as none.get()
Traceback (most recent call last):
...
ValueError: Option is empty
Option monad implements the iteration protocl
>>> o = opt([1, 2, 3]).for_each(print)
[1, 2, 3]
>>> a = opt('optional option value')
>>> for i in a:
>>> print(i)
optional option value
It supports also forwarding a value
>>> # forwarding value
>>> class A(object):
def __init__(self, x):
self.x = x
def get_x(self):
return self.x
>>> opt(A(1)).get_x().or_(0)
1
>>> opt(A(1)).get_y().or_(0)
0
A option value can be transformed into another enabling chaining operation
from fateful.monad.option import option
result = option("some").map(lambda x: len(x) * 2).or_(0)
An option can contain another option type but can be flattened
x = option(option(option(1))).flatten() # x is Some[Some[Some[int]]]
y: Some[Some[Some[int]]] = Some(Some(Some(1)))
x: Some[int] = Some(Some(Some(1))).flatten()
We can lift an function to return an option type using a decorator
from fateful.monad.option import lift_option
@lift_option
def maybe(value: int):
if value >= 2:
return 2
return None
x: Some[int] = maybe(2)
x: Empty = maybe(0)
The Result Monad is another concept from functional programming that is used to handle computations that may produce a successful result or a failure. It provides a way to encapsulate the outcome of an operation, allowing you to handle and propagate errors in a controlled manner.
The Result Monad typically has two possible states: Ok (representing a successful result) and Err (representing a failure or an error condition). The Ok state contains the successful result value, while the Err state contains information about the failure, such as an error message or an error object.
The Result Monad offers several benefits:
- Explicit error handling: It makes error handling explicit and separates the handling of successful results from error conditions.
- Propagation of errors: It allows errors to be easily propagated through a chain of operations, avoiding the need for explicit error-checking at each step.
- Composition: It enables the chaining and composition of operations on results in a concise and expressive manner.
- Error recovery: It provides mechanisms to handle and recover from errors, such as by mapping to alternative values or applying fallback strategies.
def may_fail(x: int) -> float:
return 1 / x
from fateful.monad.resut import sync_try
r: Ok[float] | Err[ZeroDivisionError] = sync_try(may_fail, ZeroDivisionError)(1)
result = sync_try(may_fail, ZeroDivisionError)(1).or_(10.0)
assert result == 1.0
result = sync_try(may_fail, ZeroDivisonError)(0).or_(10.0)
assert result == 10.0
Async result provides exactly same functionalites than regular Result monad
async def divide_async(a, b):
await asyncio.sleep(0.1)
return a / b
from fateful.monad import async_try
value = await async_try(add_async)( 1, 1).or_else(lambda: 4)
value = await async_try(add_async)(1, 1).or_(4)
divide: AsyncResult[..., float, ZeroDivisionError] = async_try(add_async, ZeroDivisionError)(1, 1)
result: Ok[float] | Err[ZeroDivisionError] = await divide..execute()
# transforming value
value = await divide.map(lambda r: r - 100).or_(0)
match (await divide.execute()):
case Ok(val):
return val
case Err(err):
raise err
case _:
raise TypeError("Never supposed to happen")
Miscellaneous functions dealing with http client functions
Wraps aiohttp for perform http call and parsing results optionnaly as json
aiohttp is a library of choice for making http requests. A good alternative is httpx but various benchmarks show that it is slightly slower.
page_content: str = await try_get("http://google.com").or_raise()
Simple subclass to return option monad when getting values implementation of list and dict
x = opt_list([1,2,3])
v: Some[int] = x.at(0)
v: Empty = x.at(12)
Miscellaneous functions dealing with json (parsing, manipulating...)
>>> from fateful.json import opt_dict
>>> i = opt_dict({'z-index': 1000})
>>> i.toto = [4, 5, 6]
>>> str(i)
'{"z-index": 1000, "toto": [4, 5, 6]}'
Notes: I found some python packages doing almost the same things. I did this essentially to learn and wanted to keep it simple.
MIT