import json import logging from http import HTTPStatus from typing import List, Optional from uuid import UUID import arrow import requests from mobilizon_reshare.config.config import get_settings from mobilizon_reshare.event.event import MobilizonEvent, EventPublicationStatus logger = logging.getLogger(__name__) class MobilizonRequestFailed(Exception): # TODO move to an error module pass def parse_location(data): if "physicalAddress" in data and data["physicalAddress"]: addr = data["physicalAddress"] return f"{addr['description']}, {addr['locality']}, {addr['region']}" elif "onlineAddress" in data and data["onlineAddress"]: return data["onlineAddress"] else: return None def parse_picture(data): return (data.get("picture", {}) or {}).get("url") def parse_event(data): return MobilizonEvent( name=data["title"], description=data.get("description", None), begin_datetime=arrow.get(data["beginsOn"]) if "beginsOn" in data else None, end_datetime=arrow.get(data["endsOn"]) if "endsOn" in data else None, mobilizon_link=data.get("url", None), mobilizon_id=UUID(data["uuid"]), thumbnail_link=parse_picture(data), location=parse_location(data), publication_time=None, status=EventPublicationStatus.WAITING, last_update_time=arrow.get(data["updatedAt"]) if "updatedAt" in data else None, ) query_future_events = """{{ group(preferredUsername: "{group}") {{ organizedEvents(page:{page}, afterDatetime:"{afterDatetime}"){{ elements {{ title, url, updatedAt, beginsOn, endsOn, options {{ showStartTime, showEndTime, }}, uuid, description, onlineAddress, physicalAddress {{ locality, description, region }}, picture {{ url }}, }}, }} }} }}""" def get_mobilizon_future_events( page: int = 1, from_date: Optional[arrow.Arrow] = None ) -> List[MobilizonEvent]: url = get_settings()["source"]["mobilizon"]["url"] query = query_future_events.format( group=get_settings()["source"]["mobilizon"]["group"], page=page, afterDatetime=from_date or arrow.now().isoformat(), ) r = requests.post(url, json={"query": query}) if r.status_code != HTTPStatus.OK: raise MobilizonRequestFailed( f"Request for events failed with code:{r.status_code}" ) response_json = r.json() logger.debug(f"Response:\n{json.dumps(response_json, indent=4)}") if "errors" in response_json: raise MobilizonRequestFailed( f"Request for events failed because of the following errors: " f"{json.dumps(response_json['errors'],indent=4)}" ) return list( map(parse_event, response_json["data"]["group"]["organizedEvents"]["elements"]) )