Successfully dealing with failures
Project description
What is failures
This library simplifies dealing with application failures especially when using multiple nested components that process some data and could fail at any point.
Often in production, the app must be robust enough against errors but still report them so we can improve it regularly,
this requires us as developers to isolate those operations (usually within try...except
blocks) then catch
any possible errors to report them and provide an alternative result (like returning None
).
This is an additional job to be done to keep achieve the production robustness.
Meanwhile, this library suggests some tools to ease this process and divides it into two main phases, the first is capturing failures through an application action and between function calls while giving them meaningful labels and optionally explicit metadata to help us understand them later and their context, the second phase is to process those collected failures with the ability to filter and choose different handling action for different type of failures.
The library focuses on two main factors, simplicity and performance, by keeping the syntax easy, clean and intuitive for a better and more readable code, and by optimizing its code to minimize the impact on your application.
Installation
failures
is available at PyPI, it requires python 3.8 or higher, to install it run the pip
command:
pip install failures
Example
This example will show the tip of the iceberg, for a more complete tutorial refer to the documentation page at documentation page.
failures
contains two main objects, Reporter
and Handler
, the first one gathers all failures and the second
on processes them.
Let's define some utilities for this example in a file named defs.py
import json
import logging
from dataclasses import dataclass
from failures import Reporter, Handler, Failure
_database = {
'692999103396568d': b'{"name": "alertCheetah2", "email": "alertCheetah2@example.com", "phone": "(+212)645-882-425"}',
'e2ddc4f976f8ccf8': b'{"name": "ardentJaguar0", "phone": "(+212)611-962-964"}',
'd097eba4d97a1ce0': b'{"name": "tautWeaver8", "email": "tautWeaver8@example.com", "phone": "(+212)683-480-745"}',
'ef0eb816db00f939': b'{"name": "awedPolenta7", "email": "awedPolenta7@example.com", "phone": "(+212)641-014-059"}',
'b54c8bea6badd65d': b'{"name": "dearCod2", "email": "dearCod2@example.com", "website": "www.dearCod2.example.com"}',
'bad-8394313d4c44': b'{"name": "panickyViper9", "ema'
}
@dataclass
class User:
id: str
name: str
email: str | None
phone: str | None
def not_found_404(failure: Failure) -> None:
try:
user_id = failure.details['id']
except KeyError:
return
logging.getLogger(failure.source).error(f"No user found with id = {user_id!r}", exc_info=failure.error)
handler = Handler((not_found_404, "*.retrieve"), print)
def get_user(user_id: str) -> User | None:
reporter = Reporter('user')
with handler:
with reporter('retrieve', id=user_id):
raw_json = _database[user_id]
with reporter('json_decode'):
user_data = json.loads(raw_json)
rep_get = reporter('get', data=user_data)
user = User(
id=user_id,
name=rep_get('name').required(lambda: user_data['name']),
email=rep_get('email').safe(user_data.__getitem__, 'email'),
phone=rep_get('phone').optional(user_data.__getitem__, 'phone')
)
handler.from_reporter(reporter)
return user
Code explained
We declared a dummy data collection _database
to act as a database, then we defined a data class User
to wrap user data.
We define then a custom handler not_found_404
to be called only when a user is not found, this handler takes
a Failure
object and returns nothing, all handlers should have this same signature.
Then we create a Handler
object used to handle application failures, we pass the custom handler under a condition
'*.retrieve'
, the condition means every label that ends with 'retrieve'
like in this case 'user.retrieve'
.
After that we define our main action get_user(user_id: str) -> User | None
that takes an id and returns a user if
it exists, we evaluate all its instructions within the handler context to capture failures, then we create the root
reporter labeled 'user'
, all following operations will be labeled user.(somthing)
, then we try to retrieve
a user by id _database[user_id]
, and as this might potentially fail, we execute it inside the reporter's context
under the label retrieve
, this context is holding the user_id
as additional detail.
Using the reporter as context manager with reporter:
ensures that the functions returns directly to the handler in
case of failure, the next code won't be executed.
The same for the next instruction, we wrap it into a labeled scope json_decode
to label it and mark it as required
step.
Then we create a User
object, this time using inline reporter context, there is three; reporter.required(...)
for mandatory operations and it's the same as with reporter: ...
, it stops and returns to the handler in case
of failure, reporter.safe(...)
does capture and keep the failure to be processed later and returns None
as
alternative result, and reporter.optional(...)
does the same but without reporting the failure.
All the three expect a callable with or without parameters to be called so reporter.safe(func, 1, 2, a=3, b=4)
executes func(1, 2, a=3, b=4)
in isolation.
After that and finally we process those reported failures and return the result.
Usage
Now let's jump into the interpreter and load our function and start calling it with different ids:
>>> from defs import get_user
>>> # Existing users
>>> get_user('692999103396568d')
User(id='692999103396568d', name='alertCheetah2', email='alertCheetah2@example.com', phone='(+212)645-882-425')
>>> get_user('e2ddc4f976f8ccf8') # Missing email reported and handled with print(failure)
Failure(source='user.get.email', error=KeyError('email'), details={'data': {'name': 'ardentJaguar0', 'phone': '(+212)611-962-964'}})
User(id='e2ddc4f976f8ccf8', name='ardentJaguar0', email=None, phone='(+212)611-962-964')
>>> get_user('d097eba4d97a1ce0')
User(id='d097eba4d97a1ce0', name='tautWeaver8', email='tautWeaver8@example.com', phone='(+212)683-480-745')
>>> get_user('ef0eb816db00f939')
User(id='ef0eb816db00f939', name='awedPolenta7', email='awedPolenta7@example.com', phone='(+212)641-014-059')
>>> get_user('b54c8bea6badd65d') # Missing phone number ignored
User(id='b54c8bea6badd65d', name='dearCod2', email='dearCod2@example.com', phone=None)
>>> # Bad json format
>>> get_user('bad-8394313d4c44') # returns None
Failure(source='user.json_decode', error=JSONDecodeError('Unterminated string starting at: line 1 column 27 (char 26)'), details={})
>>> # Non-existing users
>>> get_user('random-id')
ERROR:user.retrieve:No user found with id = 'random-id'
Traceback (most recent call last):
File "/path/to/defs.py", line 35, in get_user
KeyError: 'random-id'
Failure(source='user.retrieve', error=KeyError('random-id'), details={'id': 'random-id'})
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
File details
Details for the file failures-1.0.1.tar.gz
.
File metadata
- Download URL: failures-1.0.1.tar.gz
- Upload date:
- Size: 21.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.6
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 941bf5400266fa75ab303c1115f200c21557190938c1bdc2d373876bbc6db0c5 |
|
MD5 | feb9838b9f4d4e2e5a663950e447716e |
|
BLAKE2b-256 | 18624417826b5accaf77dbe5407d8966941d430fcac16a15656508bfb926db9d |
File details
Details for the file failures-1.0.1-py3-none-any.whl
.
File metadata
- Download URL: failures-1.0.1-py3-none-any.whl
- Upload date:
- Size: 14.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/4.0.2 CPython/3.11.6
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 0487506c626e34f649c6e33ffa13e288c31a3f5f8816359af1d49b39909a8d40 |
|
MD5 | 6613087534157c694958339a27b56a3b |
|
BLAKE2b-256 | 307268cd19ce4f44c496a0a489caeb1bf31779e6afcb1dabd24e80e00ee949aa |