|
|
|
|
|
import uuid |
|
|
import asyncio |
|
|
import logging |
|
|
from typing import Any, Optional |
|
|
|
|
|
import aiormq |
|
|
import aio_pika |
|
|
|
|
|
from config import settings |
|
|
from models import CloudEvent |
|
|
from rabbit_base import RabbitBase |
|
|
from utils import to_json, json_compress_str |
|
|
|
|
|
logger = logging.getLogger(__name__) |
|
|
|
|
|
|
|
|
class RabbitRepo(RabbitBase): |
|
|
def __init__(self, external_source: str): |
|
|
super().__init__(exchange_type_resolver=self._resolve_type) |
|
|
self._source = external_source |
|
|
|
|
|
def _resolve_type(self, exch: str) -> str: |
|
|
if exch.lower().startswith("oa."): |
|
|
return "direct" |
|
|
if hasattr(settings, 'EXCHANGE_TYPES') and settings.EXCHANGE_TYPES: |
|
|
matches = [k for k in settings.EXCHANGE_TYPES.keys() |
|
|
if exch.lower().startswith(k.lower())] |
|
|
if matches: |
|
|
return settings.EXCHANGE_TYPES[max(matches, key=len)] |
|
|
return "fanout" |
|
|
|
|
|
async def _publish_with_retry(self, exchange: str, body: bytes, routing_key: str = "") -> None: |
|
|
attempts, delay = 0, 0.5 |
|
|
while True: |
|
|
try: |
|
|
ex = await self.ensure_exchange(exchange) |
|
|
msg = aio_pika.Message( |
|
|
body=body, |
|
|
delivery_mode=aio_pika.DeliveryMode.PERSISTENT, |
|
|
) |
|
|
await ex.publish(msg, routing_key=routing_key) |
|
|
return |
|
|
except (asyncio.CancelledError, |
|
|
aiormq.exceptions.ChannelInvalidStateError, |
|
|
aiormq.exceptions.ConnectionClosed, |
|
|
aio_pika.exceptions.AMQPError, |
|
|
RuntimeError) as e: |
|
|
attempts += 1 |
|
|
logger.warning("publish failed attempt=%d exchange=%s rk=%s err=%r", |
|
|
attempts, exchange, routing_key, e) |
|
|
try: |
|
|
await self.close() |
|
|
except Exception: |
|
|
pass |
|
|
if attempts >= 5: |
|
|
logger.exception("publish giving up after %d attempts", attempts) |
|
|
raise |
|
|
await asyncio.sleep(delay) |
|
|
delay = min(delay * 2, 5.0) |
|
|
|
|
|
async def publish(self, exchange: str, obj: Any, routing_key: str = "") -> None: |
|
|
payload = obj if not hasattr(obj, "model_dump") else obj.model_dump(by_alias=True) |
|
|
evt = CloudEvent.wrap( |
|
|
event_id=str(uuid.uuid4()), |
|
|
event_type=(obj.__class__.__name__ if obj is not None else "NullOrEmpty"), |
|
|
source=self._source, |
|
|
data=payload, |
|
|
) |
|
|
body = evt.model_dump_json(exclude_none=True).encode("utf-8") |
|
|
await self._publish_with_retry(exchange, body, routing_key) |
|
|
|
|
|
async def publish_jsonz( |
|
|
self, |
|
|
exchange: str, |
|
|
obj: Any, |
|
|
routing_key: str = "", |
|
|
with_id: Optional[str] = None, |
|
|
) -> str: |
|
|
payload = obj if not hasattr(obj, "model_dump") else obj.model_dump(by_alias=True) |
|
|
datajson = to_json(payload) |
|
|
datajsonZ = json_compress_str(datajson) |
|
|
wrapped: Any = (datajsonZ, with_id) if with_id else datajsonZ |
|
|
|
|
|
evt = CloudEvent.wrap( |
|
|
event_id=str(uuid.uuid4()), |
|
|
event_type=(obj.__class__.__name__ if obj is not None else "NullOrEmpty"), |
|
|
source=self._source, |
|
|
data=wrapped, |
|
|
) |
|
|
body = evt.model_dump_json(exclude_none=True).encode("utf-8") |
|
|
await self._publish_with_retry(exchange, body, routing_key) |
|
|
return datajsonZ |
|
|
|