logo le blog invivoo blanc

Fabrique – Design Patterns

1 août 2022 | Python | 0 comments

De nombreuses applications financières doivent persister des données après leurs traitements algorithmiques. Par exemple, la gestion des portefeuilles demande d’enregistrer localement les résultats des ordres venant de différentes sources externes après le nettoyage des données dans les systèmes de stockage très variés : le système de gestion de base de données relationnelles (SGBDR), le système de gestion de base de données orienté objet (OODBMS), les documents XML, les fichiers Excel, le service Web, LDAP, etc. L’utilisateur pourrait appeler une méthode dans un module commun, qui contient tous les types d’enregistrement, en choisissant une manière de stockage désirée au gré du besoin. Néanmoins, si elle comprend tous ces types d’implémentations concrètes dont chacune propose une interface spécifique, elle ne sera pas facile à être maintenue ni étendue aux nouvelles fonctionnalités de persistance chez le client ultérieurement. Car cela viole le principe de responsabilité unique dans SOLID [1], qui stipule qu’un module, une classe ou même une méthode doit avoir une responsabilité unique et bien définie. Il ne devrait faire qu’une seule chose et n’avoir qu’une seule raison de changer. Sinon, quoi qu’elle soit dérisoire ou importante, une mise à jour sur une seule façon de stockage pourrait introduire la modification de toute la classe, voire tout le module, en risquant d’introduire les erreurs désastreuses.

Du coup en face de tel défi, le patron de conception dans la famille créatrice [2] « l’usine » : la fabrique (factory method en anglais) joue un rôle majeur et efficace en construisant tous les types d’objet qui s’occupe d’une persistance particulière à la sortie de résultat. Du fait, il y a une super classe abstraite de persistance avec plusieurs sous-classes concrètes et en fonction de l’argument d’entrée, la méthode de création de la classe factory va déterminer et retourner l’une des sous-classes. Cette conception retire la responsabilité de l’instanciation du programme chez le client aux sous-classes dans la fabrique. Sinon, au gré des contextes, elle pourrait renvoyer plusieurs fois la même instance [3, 4], ou renvoyer une sous-classe plutôt qu’un objet de ce type exact.

Conception

Le diagramme ULM illustre deux composants principaux dans le patron de conception Fabrique en Python 3, dont le 1e est un point d’accès de classe Factory qui décide et retourne l’objet final et le 2e comprend plusieurs classes concrètes dérivant d’une même interface, qui s’occupent d’instancier les objets.

Diagramme Patron de conception Fabrique Python 3 - Design patterns factory

Le client qui cherche à créer un objet accède à la classe Factory, dans laquelle la méthode statique make est appelée avec le nom de l’objet désiré. Cette méthode statique connaît tous les types d’instance des classes dérivant de l’interface en commun. Dès son analyse du besoin du client, la méthode renvoie directement la référence de l’objet demandé, lorsque cet objet est construit par la méthode de classe create implémentée dans une sous-classe concrète de l’interface abstraite.

Par conséquent, le principe de responsabilité unique est bien satisfait en faveur de la conception.

Développement

En Python 3 afin de distinguer clairement les types d’objet lors des appels dans le code, on met en place une classe énumérée PersistType qui donne tous les noms de persistance. Par exemple, dans cette application rudimentaire on a trois types de persistance dont les libraires en Python sont disponibles pour les ordres : XML [5], SGBDR [6], OODBMS [7].

Afin de simplifier la démonstration, on garde simplement les attributs basiques dans la classe Order pour les instances d’ordres. Du fait, à part les identifiants uniques pour chacun, on définit la date d’exécution, le montant d’exécution et la devise chez l’ordre. En plus, dans cet exemple le portefeuille ne contient que dix ordres exécutés il y a un jour. Les montants des ordres à la source sont générés de la façon aléatoire à la distribution uniforme [8] entre -100000,0 et 100000,0, lorsque leurs devises sont toujours en euro. Sinon, comme OODBMS demande un héritage de persistance dans sa librairie pour les ordres, on conçoit la classe OrderZo pour s’adapter à son objet d’enregistrement.

Dans un premier temps, on développe le 1e composant dans le schéma de la conception. En tant qu’un point d’accès, la classe PersistanceFactory est mise au point pour fournir les objets de persistance dont on a besoin. En sachant le nom du type de persistance, on y appelle la méthode statique make qui vérifie les conditions. Grâce à cette méthode, on peut trouver la classe de persistance pertinente à poursuivre l’instanciation concrète chez le 2e composant de la conception.

En effet, toutes ces classes de persistance ayant une méthode de classe create pour créer ses propres objets sont concrètement implémentées à partir de la interface abstraite IPersistance. Sinon, on y met en œuvre une méthode persist qui est responsable d’enregistrer les données des ordres pendant nos tests. Dès qu’une persistance instanciée sera été retournée par la méthode make chez la classe PersistanceFactory, la mise en persistance des données s’apprête à avoir lieu si la méthode persist est appelée.

# resistance_factory.py
from abc import ABC, abstractmethod
from datetime import datetime, timedelta
from random import uniform 
from xml.dom import minidom
from xml.etree import ElementTree as ET 
from ZODB import FileStorage
import BTrees.OOBTree
import enum
import persistent.list
import pprint
import sqlite3
import transaction
import ZODB

pp = pprint.PrettyPrinter(indent=4)


class PersistType(enum.Enum):
    """ An Enumerate Class to distinguish persistence """
    XML = 1
    SGBDR = 2
    OODBMS = 3


class IPersistence(ABC):
    """ An Abstract Class Interface (Persistence) """
    name = None

    @classmethod
    @abstractmethod
    def create(cls):
        """ An abstract interface method to create persistence """
        pass

    @abstractmethod
    def persist(self, orders: list):
        """
        An abstract interface method to persist orders

        Attributes
        ----------
        orders : list
        The orders to stock.
        """
        pass


class Order(object):
    def __init__(self):
        self._order_id = None
        self._pricing_date = None
        self._amount = None
        self._currency = None

    @property
    def order_id(self):
        return self._order_id

    @property
    def pricing_date(self):
        return self._pricing_date

    @property
    def amount(self):
        return self._amount

    @property
    def currency(self):
        return self._currency

    @order_id.setter
    def order_id(self, value):
        self._order_id = value

    @pricing_date.setter
    def pricing_date(self, value):
        self._pricing_date = value

    @amount.setter
    def amount(self, value):
        self._amount = value

    @currency.setter
    def currency(self, value):
        self._currency = value

    def __repr__(self):
        message = f"<{type(self).__name__}: "
        message += "; ".join(f"{k}, {v}" for k, v in vars(self).items())
        message += '>'
        return message


class OrderZo(Order, persistent.Persistent):
    def __init__(self, odr: Order):
        """
        :param odr: Order
        to adapt the ZODB persistent object
        """
        super().__init__()
        self._order_id = odr.order_id
        self._pricing_date = odr.pricing_date
        self._amount = odr.amount
        self._currency = odr.currency


class PersistenceOfXML(IPersistence):
    """ A Concrete XML Class that implements the IPersistence interface """
    name = PersistType.XML

    def __init__(self):
        self._engine = 'XML Engine'
        self.root = ET.Element('orders')

    def persist(self, orders: list):
        for order in orders:
            order_et = ET.SubElement(self.root, 'order')
            o_id = ET.SubElement(order_et, 'order_id')
            o_id.text = str(order.order_id)
            pricing_date = ET.SubElement(order_et, 'pricing_date')
            pricing_date.text = order.pricing_date.strftime(
                '%Y/%m/%d %H:%M:%S.%f')
            amount = ET.SubElement(order_et, 'amount')
            amount.text = str(order.amount)
            currency = ET.SubElement(order_et, 'currency')
            currency.text = order.currency
        xml_str = minidom.parseString(
            ET.tostring(self.root)).toprettyxml(
            indent="   "
        )
        with open("XML.xml", "w") as f:
            f.write(xml_str)
        pp.pprint(f"populating")
        pp.pprint(orders)
        pp.pprint(f"by {self._engine}...")

    @classmethod
    def create(cls) -> IPersistence:
        print('XML persistence will be ready...')
        return cls()


class PersistenceOfSGBDR(IPersistence):
    """ A Concrete SGBDR Class that implements the IPersistence interface """
    name = PersistType.SGBDR

    def __init__(self):
        self._engine = 'SGBDR Engine'
        self.conn = None
        try:
            self.conn = sqlite3.connect('SGBDR.db')
        except Exception as e:
            print(e)
        cur = self.conn.cursor()
        # Drop any existing table with the same name
        try:
            drop_table = "DROP TABLE ORDERS"
            cur.execute(drop_table)
        except Exception as e:
            print(e)
        # Create a table in the disk file based database
        create_table = (f"CREATE TABLE IF NOT EXISTS ORDERS("
                        f"ORDER_ID INTEGER PRIMARY KEY, "
                        f"PRICING_DATE DATETIME NOT NULL, "
                        f"AMOUNT NUMERIC(10,8), "
                        f"CURRENCY CHARACTER(3))")

        cur.execute(create_table)
        # commit the table to db
        self.conn.commit()

    def persist(self, orders: list):
        cur = self.conn.cursor()
        for order in orders:
            sql = '''
            INSERT INTO ORDERS(ORDER_ID, PRICING_DATE, AMOUNT, CURRENCY)
            VALUES(?,?,?,?)
            '''
            cur.execute(sql, (order.order_id,
                              order.pricing_date,
                              order.amount,
                              order.currency))
        self.conn.commit()
        self.conn.close()
        pp.pprint(f"populating")
        pp.pprint(orders)
        pp.pprint(f"by {self._engine} ending in {cur.lastrowid}...")

    @classmethod
    def create(cls) -> IPersistence:
        print('SGBDR persistence will be ready...')
        return cls()


class PersistenceOfOODBMS(IPersistence):
    """ A Concrete OODBMS Class that implements the IPersistence interface """
    name = PersistType.OODBMS

    def __init__(self):
        self._engine = 'OODBMS Engine'
        storage = FileStorage.FileStorage('OODBMS.fs')
        db = ZODB.DB(storage)
        connection = db.open()
        self.root = connection.root

    def persist(self, orders: list):
        self.root.orders = BTrees.OOBTree.BTree()
        self.root.orders = [OrderZo(z) for z in orders]
        transaction.commit()
        pp.pprint(f"populating")
        pp.pprint(orders)
        pp.pprint(f"by {self._engine}...")

    @classmethod
    def create(cls) -> IPersistence:
        print('OODBMS persistence will be ready...')
        return cls()


class PersistenceFactory(object):
    """ The Persistence Factory Class """

    @staticmethod
    def make(name) -> IPersistence:
        """ A static method to get a concrete product """
        if name == PersistType.XML:
            return PersistenceOfXML.create()
        if name == PersistType.SGBDR:
            return PersistenceOfSGBDR.create()
        if name == PersistType.OODBMS:
            return PersistenceOfOODBMS.create()


if __name__ == '__main__':
    # The Client
    for i, t in enumerate(PersistType):
        print(f"In Phase {i}: ")
        persistence = PersistenceFactory().make(t)
        order_samples = []
        for order_id in range(10):
            o = Order()
            o.order_id = order_id
            o.pricing_date = datetime.today() - timedelta(days=1)
            o.amount = uniform(-10000.0, 10000.0)
            o.currency = 'EUR'
            order_samples.append(o)
        persistence.persist(order_samples)
        assert persistence.name is t

Les tests unitaires se valident par les types des objets créés au gré du besoin, tandis que les affichages des différentes phases de persistance pour dix ordres sont mis en lumière en-dessous :

In Phase 0: 
XML persistence will be ready...
'populating'
[   <Order: _order_id, 0; _pricing_date, 2022-03-12 22:57:18.577233; _amount, 8682.064027577868; _currency, EUR>,
    <Order: _order_id, 1; _pricing_date, 2022-03-12 22:57:18.577271; _amount, 1178.6677560764638; _currency, EUR>,
    <Order: _order_id, 2; _pricing_date, 2022-03-12 22:57:18.577280; _amount, 877.1820266699797; _currency, EUR>,
    <Order: _order_id, 3; _pricing_date, 2022-03-12 22:57:18.577285; _amount, 3557.339177963937; _currency, EUR>,
    <Order: _order_id, 4; _pricing_date, 2022-03-12 22:57:18.577290; _amount, -8528.180149274358; _currency, EUR>,
    <Order: _order_id, 5; _pricing_date, 2022-03-12 22:57:18.577294; _amount, -4527.9443459321665; _currency, EUR>,
    <Order: _order_id, 6; _pricing_date, 2022-03-12 22:57:18.577299; _amount, -6062.7853847754905; _currency, EUR>,
    <Order: _order_id, 7; _pricing_date, 2022-03-12 22:57:18.577303; _amount, 1449.8770822999832; _currency, EUR>,
    <Order: _order_id, 8; _pricing_date, 2022-03-12 22:57:18.577308; _amount, -1708.4810066747305; _currency, EUR>,
    <Order: _order_id, 9; _pricing_date, 2022-03-12 22:57:18.577317; _amount, 5091.879830019116; _currency, EUR>]
'by XML Engine...'
In Phase 1: 
SGBDR persistence will be ready...
'populating'
[   <Order: _order_id, 0; _pricing_date, 2022-03-12 22:57:18.588848; _amount, 8990.32508880787; _currency, EUR>,
    <Order: _order_id, 1; _pricing_date, 2022-03-12 22:57:18.588875; _amount, -7029.962047408302; _currency, EUR>,
    <Order: _order_id, 2; _pricing_date, 2022-03-12 22:57:18.588884; _amount, -6521.53303624182; _currency, EUR>,
    <Order: _order_id, 3; _pricing_date, 2022-03-12 22:57:18.588891; _amount, 8693.092444597798; _currency, EUR>,
    <Order: _order_id, 4; _pricing_date, 2022-03-12 22:57:18.588897; _amount, 9584.87989883318; _currency, EUR>,
    <Order: _order_id, 5; _pricing_date, 2022-03-12 22:57:18.588903; _amount, -8554.88392848472; _currency, EUR>,
    <Order: _order_id, 6; _pricing_date, 2022-03-12 22:57:18.588908; _amount, 3109.2992332277263; _currency, EUR>,
    <Order: _order_id, 7; _pricing_date, 2022-03-12 22:57:18.588914; _amount, -7536.156871450268; _currency, EUR>,
    <Order: _order_id, 8; _pricing_date, 2022-03-12 22:57:18.588919; _amount, 8377.004299016604; _currency, EUR>,
    <Order: _order_id, 9; _pricing_date, 2022-03-12 22:57:18.588925; _amount, 4520.661799666674; _currency, EUR>]
'by SGBDR Engine ending in 9...'
In Phase 2: 
OODBMS persistence will be ready...
'populating'
[   <Order: _order_id, 0; _pricing_date, 2022-03-12 22:57:18.595025; _amount, 4467.882449831253; _currency, EUR>,
    <Order: _order_id, 1; _pricing_date, 2022-03-12 22:57:18.595049; _amount, 5956.087440158884; _currency, EUR>,
    <Order: _order_id, 2; _pricing_date, 2022-03-12 22:57:18.595058; _amount, 7233.788601777396; _currency, EUR>,
    <Order: _order_id, 3; _pricing_date, 2022-03-12 22:57:18.595065; _amount, 8032.730099866159; _currency, EUR>,
    <Order: _order_id, 4; _pricing_date, 2022-03-12 22:57:18.595072; _amount, 118.40014380573666; _currency, EUR>,
    <Order: _order_id, 5; _pricing_date, 2022-03-12 22:57:18.595079; _amount, 2196.295149587426; _currency, EUR>,
    <Order: _order_id, 6; _pricing_date, 2022-03-12 22:57:18.595087; _amount, 580.9771003834958; _currency, EUR>,
    <Order: _order_id, 7; _pricing_date, 2022-03-12 22:57:18.595093; _amount, 9547.969637860293; _currency, EUR>,
    <Order: _order_id, 8; _pricing_date, 2022-03-12 22:57:18.595100; _amount, 3679.870172622952; _currency, EUR>,
    <Order: _order_id, 9; _pricing_date, 2022-03-12 22:57:18.595108; _amount, -4532.329828675801; _currency, EUR>]
'by OODBMS Engine...'

Comme tous les types de persistance sont testés par trois itérations dans le programme, les objets de persistance XML, SGBDR et OODBMS sont instanciés l’un après l’autre avant leurs enregistrements de données des ordres.

Le ficher XML.xml stocke les ordres ainsi :

<?xml version="1.0" ?>
<orders>
   <order>
      <order_id>0</order_id>
      <pricing_date>2022/03/12 22:57:18.577233</pricing_date>
      <amount>8682.064027577868</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>1</order_id>
      <pricing_date>2022/03/12 22:57:18.577271</pricing_date>
      <amount>1178.6677560764638</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>2</order_id>
      <pricing_date>2022/03/12 22:57:18.577280</pricing_date>
      <amount>877.1820266699797</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>3</order_id>
      <pricing_date>2022/03/12 22:57:18.577285</pricing_date>
      <amount>3557.339177963937</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>4</order_id>
      <pricing_date>2022/03/12 22:57:18.577290</pricing_date>
      <amount>-8528.180149274358</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>5</order_id>
      <pricing_date>2022/03/12 22:57:18.577294</pricing_date>
      <amount>-4527.9443459321665</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>6</order_id>
      <pricing_date>2022/03/12 22:57:18.577299</pricing_date>
      <amount>-6062.7853847754905</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>7</order_id>
      <pricing_date>2022/03/12 22:57:18.577303</pricing_date>
      <amount>1449.8770822999832</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>8</order_id>
      <pricing_date>2022/03/12 22:57:18.577308</pricing_date>
      <amount>-1708.4810066747305</amount>
      <currency>EUR</currency>
   </order>
   <order>
      <order_id>9</order_id>
      <pricing_date>2022/03/12 22:57:18.577317</pricing_date>
      <amount>5091.879830019116</amount>
      <currency>EUR</currency>
   </order>
</orders>

Dans la base de Sqlite3, nous pouvons saisir toutes les données dans la table ORDERS :

sqlite> sqlite> SELECT * FROM ORDERS;
0|2022-03-12 22:57:18.588848|8990.32508880787|EUR
1|2022-03-12 22:57:18.588875|-7029.9620474083|EUR
2|2022-03-12 22:57:18.588884|-6521.53303624182|EUR
3|2022-03-12 22:57:18.588891|8693.0924445978|EUR
4|2022-03-12 22:57:18.588897|9584.87989883318|EUR
5|2022-03-12 22:57:18.588903|-8554.88392848472|EUR
6|2022-03-12 22:57:18.588908|3109.29923322773|EUR
7|2022-03-12 22:57:18.588914|-7536.15687145027|EUR
8|2022-03-12 22:57:18.588919|8377.0042990166|EUR
9|2022-03-12 22:57:18.588925|4520.66179966667|EUR

Les résultats gérés chez OODBMS peuvent être décodés ainsi :

{'_order_id': 0, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595025), '_amount': 4467.882449831253, '_currency': 'EUR'}
{'_order_id': 1, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595049), '_amount': 5956.087440158884, '_currency': 'EUR'}
{'_order_id': 2, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595058), '_amount': 7233.788601777396, '_currency': 'EUR'}
{'_order_id': 3, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595065), '_amount': 8032.730099866159, '_currency': 'EUR'}
{'_order_id': 4, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595072), '_amount': 118.40014380573666, '_currency': 'EUR'}
{'_order_id': 5, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595079), '_amount': 2196.295149587426, '_currency': 'EUR'}
{'_order_id': 6, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595087), '_amount': 580.9771003834958, '_currency': 'EUR'}
{'_order_id': 7, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595093), '_amount': 9547.969637860293, '_currency': 'EUR'}
{'_order_id': 8, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595100), '_amount': 3679.870172622952, '_currency': 'EUR'}
{'_order_id': 9, '_pricing_date': datetime.datetime(2022, 3, 12, 22, 57, 18, 595108), '_amount': -4532.329828675801, '_currency': 'EUR'}

De cette façon, on réussit à réaliser les deux composants principaux dans le patron de conception Fabrique en Python 3, qui surmonte la violation du principe de responsabilité unique pour la persistance des ordres chez l’utilisateur montrée au début de l’article. Désormais on pourrait enrichir à l’aise autant de types d’objet que l’on veut aux contextes plus compliqués, tout en ne rajoutant les conditions de justification et les classes dérivées que dans les deux composants respectivement, lorsque les anciennes fonctionnalités opérationnelles ne risquent pas d’être impactées dans les applications.

Conclusion

Dans cet article, on met en évidence la nécessité du patron de conception Fabrique contre l’issue de persistance dans notre contexte en question. Son diagramme ULM comprend deux principaux composants qui respectent le principe de responsabilité unique. Ce patron de conception créateur est implémenté et testé en Python 3 pour une application rudimentaire, lorsque l’enrichissement des nouvelles créations pourrait passer à l’échelle dans le code en sécurité.

Références

[1] https://www.invivoo.com/lart-clean-code-environnement-java/
[2] https://www.invivoo.com/design-patterns-patrons-conception/
[3] https://www.invivoo.com/singleton-design-patterns-part1/
[4] https://www.invivoo.com/singletons-design-patterns-partie-2/
[5] https://www.tutorialspoint.com/python/python_xml_processing.htm
[6] https://docs.python.org/3/library/sqlite3.html
[7] https://zodb.org/en/latest/
[8] https://docs.python.org/fr/3/library/random.html