Metadata-Version: 2.1
Name: petisco
Version: 0.16.0
Summary: Petisco is a framework for helping Python developers to build clean Applications
Home-page: https://github.com/alice-biometrics/petisco
Author: ALiCE Biometrics
Author-email: support@alicebiometrics.com
License: MIT
Keywords: DDD,Use Case,Clean Architecture,REST,Applications
Platform: UNKNOWN
Classifier: License :: OSI Approved :: MIT License
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3
Description-Content-Type: text/markdown
Requires-Dist: meiga (>=1.2.3)
Requires-Dist: dataclasses-json (==0.3.8)
Requires-Dist: pyjwt (>=1.7.1)
Requires-Dist: cryptography (>=2.1.4)
Requires-Dist: deprecation
Requires-Dist: dataclasses (==0.7) ; python_version < "3.7"
Requires-Dist: backports-datetime-fromisoformat (>=1.0.0) ; python_version < "3.7"
Provides-Extra: flask
Requires-Dist: connexion (==2.6.0) ; extra == 'flask'
Requires-Dist: connexion[swagger-ui] ; extra == 'flask'
Requires-Dist: Flask-Cors (==3.0.7) ; extra == 'flask'
Provides-Extra: gunicorn
Requires-Dist: gunicorn ; extra == 'gunicorn'
Requires-Dist: json-logging-py (==0.2) ; extra == 'gunicorn'
Provides-Extra: rabbitmq
Requires-Dist: pika (==1.1.0) ; extra == 'rabbitmq'
Provides-Extra: redis
Requires-Dist: redis (>=3.3.11) ; extra == 'redis'
Requires-Dist: fakeredis (>=1.0.5) ; extra == 'redis'
Provides-Extra: sqlalchemy
Requires-Dist: sqlalchemy (>=1.3.11) ; extra == 'sqlalchemy'
Requires-Dist: sqlalchemy-utils ; extra == 'sqlalchemy'
Requires-Dist: PyMySQL (==0.9.2) ; extra == 'sqlalchemy'

# petisco :cookie:  [![version](https://img.shields.io/github/release/alice-biometrics/petisco/all.svg)](https://github.com/alice-biometrics/petisco/releases) [![ci](https://github.com/alice-biometrics/petisco/workflows/ci/badge.svg)](https://github.com/alice-biometrics/petisco/actions) [![pypi](https://img.shields.io/pypi/dm/petisco)](https://pypi.org/project/petisco/)

<img src="https://github.com/alice-biometrics/custom-emojis/blob/master/images/alice_header.png" width=auto>

Petisco is a framework for helping Python developers to build clean Applications in Python.

:warning: disclaimer: not stable yet


## Table of Contents
- [Installation :computer:](#installation-computer)
- [Getting Started :chart_with_upwards_trend:](#getting-started-chart_with_upwards_trend)
    * [ApplicationConfig](#applicationconfig)
    * [Handlers](#handlers)
    * [Model your Domain](#model-your-domain)
      - [Value Objects](#value-objects)
      - [Events](#events)
      - [Aggregate Root](#aggregate-root)
- [Extras](#extras)
- [Contact :mailbox_with_mail:](#contact-mailbox_with_mail)


## Installation :computer:

```console
pip install petisco
```

Installation with Extras 

```console
pip install petisco[flask]
pip install petisco[sqlalchemy]
pip install petisco[redis]
pip install petisco[rabbitmq]
pip install petisco[flask,sqlalchemy,redis,rabbitmq]
```

## Getting Started :chart_with_upwards_trend:	

**petisco** provides us some sort of interfaces and decorators to help on the development of clean architecture Applications.

### ApplicationConfig

Before to run your app, you should configure it with `ApplicationConfig` object.

`ApplicationConfig` is a singleton with the following parameters:

    Parameters
    ----------
    app_name
        Application name
    mode
        DeploymentMode define the toy_app mode of execution. If you're mapping services and repositories, please
        check given mode is mapped in services_mode_mapper and repositories_mode_mapper
    logger
        Pre configured logger
    config_dependencies
        Callable function to configure dependencies (e.g configure credentials in order to connect with a thrid-party
        toy_app.
    config_persistence
        Callable function to configure toy_app persistence (e.g configure a database)
    services_mode_mapper
        A dictionary to map DeploymentMode with a service provider function. This is used as a dependency injector
    repositories_mode_mapper
        A dictionary to map DeploymentMode with a repository provider function. This is used as a dependency injector
    event_manager
        A IEventManager valid implementation
    options
        A dictionary with specific toy_app options


Check a configuration example in the [Integration Tests](tests/integration/toy_app/application_setup.py)


### Handlers

**petisco** implement a sort of decorator to handle common behaviour of application elements.

#### Controller Handler

Add it to your entry point controller and manage the behaviour:

```python
    from petisco import controller_handler
    from meiga import Success

    @controller_handler()
    def my_controller(headers=None):
        return Success("Hello Petisco")
```
*controller_handler parameters:*

    Parameters
    ----------
    app_name
        Application name
    logger
        A ILogger implementation. Default NotImplementedLogger
    event_config
        EventConfig object. Here, you can define event management.
    jwt_config
        JwtConfig object. Here, you can define how to deal with JWT Tokens
    success_handler
        Handler to deal with Success Results
    error_handler
        Handler to deal with Failure Results
    correlation_id_provider
        Injectable function to provide correlation_id. By default is used flask_correlation_id_provider
    headers_provider
        Injectable function to provide headers. By default is used headers_provider
    logging_types_blacklist
        Logging Blacklist. Object of defined Type will not be logged. By default ( [bytes] ) bytes object won't be logged.


### Model your Domain


#### Value Objects

Extend `ValueObject` to model your Value Objects.

Find some examples in [petisco/domain/value_objects](petisco/domain/value_objects)

#### Events

Extend `Event` to model your domain events.

```python
from petisco import Event, UserId, Name

class UserCreated(Event):
    user_id: UserId
    name: Name

    def __init__(self, user_id: UserId, name: Name):
        self.user_id = user_id
        self.name = name
        super().__init__()
```

#### Aggregate Root

Extend `AggregateRoot` to model your Aggregate Roots

```python
from petisco import AggregateRoot, UserId, Name
from my_code import UserCreated

class User(AggregateRoot):

    def __init__(self, name: Name, user_id: UserId):
        self.name = name
        self.user_id = user_id
        super().__init__()

    @staticmethod
    def create(name: Name):
        user = User(name, UserId.generate())
        user.record(UserCreated(user.user_id, user.name))
        return user
```

Use semantic constructors and `record` domain `Event`s very easy.

```python 
user = User.create(Name("Petisco"))
events = user.pull_domain_events() # Events ready to be published
```

#### Extras

###### RabbitMQ

To test RabbitEventManager you need to run locally a RabbitMQ application, otherwise related test will be skipped.
Please, check the official doc here: https://www.rabbitmq.com/download.html

With docker

```console
docker run -it --rm --name rabbitmq -p 5672:5672 -p 15672:15672 rabbitmq:3-management
```

How to use the RabbitMQEventManager:

```python
from time import sleep

from pika import ConnectionParameters
from petisco import Event, RabbitMQEventManager, UserId


class UserCreated(Event):
    user_id: UserId

    def __init__(self, user_id: UserId):
        self.user_id = user_id
        super().__init__()


def callback(ch, method, properties, body):
    event = Event.from_json(body)
    print(f" [x] Received {event}")
    # do your stuff here
    ok = True
    if ok:
        ch.basic_ack(delivery_tag=method.delivery_tag)
    else:
        ch.basic_nack(delivery_tag=method.delivery_tag)


topic = "petisco"
event_manager = RabbitMQEventManager(
    connection_parameters=ConnectionParameters(host="localhost"),
    subscribers={topic: callback},
)

event_manager.publish(
    topic, UserCreated(user_id=UserId.generate())
)

sleep(0.5)  # wait for the callback

event_manager.unsubscribe_all()
```

## Development

### Using lume

```console
pip install lume
```

Then:

```console 
lume -install -all
```


## Contact :mailbox_with_mail:

support@alicebiometrics.com


