refacto: reorganize POC into clean architecture for modularity
This commit is contained in:
parent
9dd53b1b24
commit
0830c2f182
14 changed files with 194 additions and 104 deletions
0
app/consumer/src/adapters/__init__.py
Normal file
0
app/consumer/src/adapters/__init__.py
Normal file
50
app/consumer/src/adapters/mqtt.py
Normal file
50
app/consumer/src/adapters/mqtt.py
Normal file
|
|
@ -0,0 +1,50 @@
|
|||
import json
|
||||
import logging
|
||||
from typing import Callable
|
||||
|
||||
import paho.mqtt.client as mqtt
|
||||
from paho.mqtt.client import ConnectFlags
|
||||
from paho.mqtt.enums import CallbackAPIVersion
|
||||
from paho.mqtt.properties import Properties
|
||||
from paho.mqtt.reasoncodes import ReasonCode
|
||||
|
||||
from entities import UplinkEvent
|
||||
from ports import MessageBroker
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
class PahoMqttBroker(MessageBroker):
|
||||
def __init__(self, host: str, port: int, topic: str):
|
||||
self._host = host
|
||||
self._port = port
|
||||
self._topic = topic
|
||||
|
||||
def start(self, on_uplink: Callable[[UplinkEvent], None]) -> None:
|
||||
_client = mqtt.Client(CallbackAPIVersion.VERSION2)
|
||||
|
||||
def _on_connect(
|
||||
client: mqtt.Client,
|
||||
userdata: None,
|
||||
flags: ConnectFlags,
|
||||
reason_code: ReasonCode,
|
||||
props: Properties | None,
|
||||
) -> None:
|
||||
log.info("MQTT connecté (code=%s)", reason_code)
|
||||
client.subscribe(self._topic, qos=1)
|
||||
|
||||
def _on_message(client: mqtt.Client, userdata: None, msg: mqtt.MQTTMessage) -> None:
|
||||
try:
|
||||
body = json.loads(msg.payload)
|
||||
dev_eui = body["deviceInfo"]["devEui"]
|
||||
pulse_count = int(body["object"]["pulse_count"])
|
||||
|
||||
on_uplink(UplinkEvent(dev_eui=dev_eui, pulse_count=pulse_count))
|
||||
except Exception as e:
|
||||
log.exception("Erreur parsing message MQTT : %s", e)
|
||||
|
||||
_client.on_connect = _on_connect
|
||||
_client.on_message = _on_message
|
||||
_client.reconnect_delay_set(min_delay=1, max_delay=30)
|
||||
_client.connect(self._host, self._port, keepalive=60)
|
||||
_client.loop_forever()
|
||||
56
app/consumer/src/adapters/postgres.py
Normal file
56
app/consumer/src/adapters/postgres.py
Normal file
|
|
@ -0,0 +1,56 @@
|
|||
import logging
|
||||
import time
|
||||
|
||||
import psycopg2
|
||||
from psycopg2.extensions import connection
|
||||
from ports import DeviceRepository, ReadingRepository
|
||||
|
||||
log = logging.getLogger(__name__)
|
||||
|
||||
|
||||
def connect(uri: str) -> connection:
|
||||
for _ in range(10):
|
||||
try:
|
||||
conn = psycopg2.connect(uri)
|
||||
conn.autocommit = True
|
||||
log.info("PostgreSQL connecté")
|
||||
return conn
|
||||
except Exception as e:
|
||||
log.warning("Attente PostgreSQL... (%s)", e)
|
||||
time.sleep(3)
|
||||
raise RuntimeError("Impossible de se connecter à PostgreSQL")
|
||||
|
||||
|
||||
class PgDeviceRepository(DeviceRepository):
|
||||
def __init__(self, conn: connection):
|
||||
self._conn = conn
|
||||
|
||||
def get_or_create_device_id(self, dev_eui: str) -> str:
|
||||
with self._conn.cursor() as cur:
|
||||
cur.execute(
|
||||
"""
|
||||
INSERT INTO device (device_eui)
|
||||
VALUES (%s)
|
||||
ON CONFLICT (device_eui) DO NOTHING
|
||||
""",
|
||||
(dev_eui,),
|
||||
)
|
||||
cur.execute(
|
||||
"SELECT device_id FROM device WHERE device_eui = %s", (dev_eui,)
|
||||
)
|
||||
return str(cur.fetchone()[0])
|
||||
|
||||
|
||||
class PgReadingRepository(ReadingRepository):
|
||||
def __init__(self, conn: connection):
|
||||
self._conn = conn
|
||||
|
||||
def insert_reading(self, device_id: str, pulse_count: int) -> None:
|
||||
with self._conn.cursor() as cur:
|
||||
cur.execute(
|
||||
"""
|
||||
INSERT INTO reading (device_id, date, pulses)
|
||||
VALUES (%s, NOW(), %s)
|
||||
""",
|
||||
(device_id, pulse_count),
|
||||
)
|
||||
Loading…
Add table
Add a link
Reference in a new issue