68 lines
2.2 KiB
Python
68 lines
2.2 KiB
Python
from typing import Any
|
|
from typing import Dict
|
|
from typing import List
|
|
from typing import Optional
|
|
|
|
import requests
|
|
|
|
from .errors import RecursionLimitExceededError
|
|
from .errors import UnexpectedActivityTypeError
|
|
|
|
|
|
def _do_req(url: str, headers: Dict[str, str]) -> Dict[str, Any]:
|
|
resp = requests.get(url, headers=headers)
|
|
resp.raise_for_status()
|
|
return resp.json()
|
|
|
|
|
|
def parse_collection(
|
|
payload: Optional[Dict[str, Any]] = None,
|
|
url: Optional[str] = None,
|
|
user_agent: Optional[str] = None,
|
|
level: int = 0,
|
|
do_req: Any = _do_req,
|
|
) -> List[str]:
|
|
"""Resolve/fetch a `Collection`/`OrderedCollection`."""
|
|
if level > 3:
|
|
raise RecursionLimitExceededError('recursion limit exceeded')
|
|
|
|
# Go through all the pages
|
|
headers = {'Accept': 'application/activity+json'}
|
|
if user_agent:
|
|
headers['User-Agent'] = user_agent
|
|
|
|
out: List[str] = []
|
|
if url:
|
|
payload = do_req(url, headers)
|
|
if not payload:
|
|
raise ValueError('must at least prove a payload or an URL')
|
|
|
|
if payload['type'] in ['Collection', 'OrderedCollection']:
|
|
if 'orderedItems' in payload:
|
|
return payload['orderedItems']
|
|
if 'items' in payload:
|
|
return payload['items']
|
|
if 'first' in payload:
|
|
if 'orderedItems' in payload['first']:
|
|
out.extend(payload['first']['orderedItems'])
|
|
if 'items' in payload['first']:
|
|
out.extend(payload['first']['items'])
|
|
n = payload['first'].get('next')
|
|
if n:
|
|
out.extend(parse_collection(url=n, user_agent=user_agent, level=level+1, do_req=do_req))
|
|
return out
|
|
|
|
while payload:
|
|
if payload['type'] in ['CollectionPage', 'OrderedCollectionPage']:
|
|
if 'orderedItems' in payload:
|
|
out.extend(payload['orderedItems'])
|
|
if 'items' in payload:
|
|
out.extend(payload['items'])
|
|
n = payload.get('next')
|
|
if n is None:
|
|
break
|
|
payload = do_req(n, headers)
|
|
else:
|
|
raise UnexpectedActivityTypeError('unexpected activity type {}'.format(payload['type']))
|
|
|
|
return out
|