2023-03-07 10:26:45 +03:00
|
|
|
from asyncio import exceptions, sleep
|
2022-08-24 08:09:41 +03:00
|
|
|
from json import loads
|
2022-11-02 20:11:38 +03:00
|
|
|
from logging import getLogger
|
2022-08-24 08:09:41 +03:00
|
|
|
from typing import AsyncGenerator
|
|
|
|
from urllib.parse import urlencode
|
2022-08-24 08:28:18 +03:00
|
|
|
from mastoposter.types import Status
|
2022-08-24 08:09:41 +03:00
|
|
|
|
2022-11-02 20:11:38 +03:00
|
|
|
logger = getLogger("sources")
|
|
|
|
|
2022-08-24 08:09:41 +03:00
|
|
|
|
2022-08-26 18:37:36 +03:00
|
|
|
async def websocket_source(
|
2023-03-07 10:26:45 +03:00
|
|
|
url: str, reconnect: bool = False, reconnect_delay: float = 1.0, **params
|
2022-08-26 18:37:36 +03:00
|
|
|
) -> AsyncGenerator[Status, None]:
|
2022-08-24 08:09:41 +03:00
|
|
|
from websockets.client import connect
|
2022-08-26 18:37:36 +03:00
|
|
|
from websockets.exceptions import WebSocketException
|
2022-08-24 08:09:41 +03:00
|
|
|
|
|
|
|
url = f"{url}?" + urlencode({"stream": "list", **params})
|
2022-08-26 18:37:36 +03:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
async with connect(url) as ws:
|
2022-08-27 14:27:42 +03:00
|
|
|
while (msg := await ws.recv()) is not None:
|
2022-08-26 18:37:36 +03:00
|
|
|
event = loads(msg)
|
|
|
|
if "error" in event:
|
|
|
|
raise Exception(event["error"])
|
|
|
|
if event["event"] == "update":
|
|
|
|
yield Status.from_dict(loads(event["payload"]))
|
2022-11-02 20:11:38 +03:00
|
|
|
else:
|
|
|
|
logger.warn("unknown event type %r", event["event"])
|
|
|
|
logger.debug("data: %r", event)
|
|
|
|
except (
|
|
|
|
WebSocketException,
|
|
|
|
TimeoutError,
|
|
|
|
exceptions.TimeoutError,
|
2023-03-19 20:09:43 +03:00
|
|
|
ConnectionError,
|
2022-11-02 20:11:38 +03:00
|
|
|
) as e:
|
2022-08-26 18:37:36 +03:00
|
|
|
if not reconnect:
|
|
|
|
raise
|
2022-11-02 20:11:38 +03:00
|
|
|
else:
|
|
|
|
logger.warn("%r caught, reconnecting", e)
|
2023-03-07 10:26:45 +03:00
|
|
|
await sleep(reconnect_delay)
|
2022-11-02 20:11:38 +03:00
|
|
|
else:
|
|
|
|
logger.info(
|
|
|
|
"WebSocket closed connection without any errors, "
|
|
|
|
"but we're not done yet"
|
|
|
|
)
|
2023-03-07 10:26:45 +03:00
|
|
|
await sleep(reconnect_delay)
|