from dataclasses import dataclass, asdict from enum import IntEnum from typing import Optional from uuid import UUID import arrow from jinja2 import Template from mobilizon_reshare.config.config import get_settings from mobilizon_reshare.models.event import Event from mobilizon_reshare.models.publication import PublicationStatus, Publication class EventPublicationStatus(IntEnum): WAITING = 1 FAILED = 2 COMPLETED = 3 PARTIAL = 4 @dataclass class MobilizonEvent: """Class representing an event retrieved from Mobilizon.""" name: str description: Optional[str] begin_datetime: arrow.Arrow end_datetime: arrow.Arrow mobilizon_link: str mobilizon_id: UUID last_update_time: arrow.Arrow thumbnail_link: Optional[str] = None location: Optional[str] = None publication_time: Optional[dict[str, arrow.Arrow]] = None status: EventPublicationStatus = EventPublicationStatus.WAITING def __post_init__(self): assert self.begin_datetime.tzinfo == self.end_datetime.tzinfo assert self.begin_datetime < self.end_datetime if self.publication_time is None: self.publication_time = {} if self.publication_time: assert self.status in [ EventPublicationStatus.COMPLETED, EventPublicationStatus.PARTIAL, EventPublicationStatus.FAILED, ] def _fill_template(self, pattern: Template) -> str: config = get_settings() return pattern.render(locale=config["locale"], **asdict(self)) def format(self, pattern: Template) -> str: return self._fill_template(pattern) @classmethod def from_model(cls, event: Event): publication_status = cls._compute_event_status(list(event.publications)) publication_time = {} for pub in event.publications: if publication_status != EventPublicationStatus.WAITING: assert pub.timestamp is not None publication_time[pub.publisher.name] = arrow.get(pub.timestamp).to( "local" ) return cls( name=event.name, description=event.description, begin_datetime=arrow.get(event.begin_datetime).to("local"), end_datetime=arrow.get(event.end_datetime).to("local"), mobilizon_link=event.mobilizon_link, mobilizon_id=event.mobilizon_id, thumbnail_link=event.thumbnail_link, location=event.location, publication_time=publication_time, status=publication_status, last_update_time=arrow.get(event.last_update_time).to("local"), ) def to_model(self, db_id: Optional[UUID] = None) -> Event: kwargs = { "name": self.name, "description": self.description, "mobilizon_id": self.mobilizon_id, "mobilizon_link": self.mobilizon_link, "thumbnail_link": self.thumbnail_link, "location": self.location, "begin_datetime": self.begin_datetime.astimezone( self.begin_datetime.tzinfo ), "end_datetime": self.end_datetime.astimezone(self.end_datetime.tzinfo), "last_update_time": self.last_update_time.astimezone( self.last_update_time.tzinfo ), } if db_id is not None: kwargs.update({"id": db_id}) return Event(**kwargs) @staticmethod def _compute_event_status( publications: list[Publication], ) -> EventPublicationStatus: if not publications: return EventPublicationStatus.WAITING unique_statuses: set[PublicationStatus] = set( pub.status for pub in publications ) if unique_statuses == { PublicationStatus.COMPLETED, PublicationStatus.FAILED, }: return EventPublicationStatus.PARTIAL elif len(unique_statuses) == 1: return EventPublicationStatus[unique_statuses.pop().name] raise ValueError(f"Illegal combination of PublicationStatus: {unique_statuses}")