HPI/my/vk/favorites.py
2024-08-10 17:46:30 +01:00

85 lines
1.9 KiB
Python

# todo: uses my private export script?, timezone
from dataclasses import dataclass
from datetime import datetime, timezone
import json
from typing import Iterator, Iterable, Optional
from my.core import Json, datetime_aware, stat, Stats
from my.core.error import Res
from my.config import vk as config # type: ignore[attr-defined]
@dataclass
class Favorite:
dt: datetime_aware
title: str
url: Optional[str]
text: str
skip = (
'graffiti',
'poll',
'note', # TODO could be useful..
'doc',
'audio',
'photo',
'album',
'video',
'page',
)
def parse_fav(j: Json) -> Favorite:
# TODO copy_history??
url = None
title = '' # TODO ???
atts = j.get('attachments', [])
for a in atts:
if any(k in a for k in skip):
continue
link = a['link']
title = link['title']
url = link['url']
# TODOlink['description'] ?
# TODO would be nice to include user
return Favorite(
dt=datetime.fromtimestamp(j['date'], tz=timezone.utc),
title=title,
url=url,
text=j['text'],
)
def _iter_favs() -> Iterator[Res]:
jj = json.loads(config.favs_file.read_text())
for j in jj:
try:
yield parse_fav(j)
except Exception as e:
ex = RuntimeError(f"Error while processing\n{j}")
ex.__cause__ = e
yield ex
def favorites() -> Iterable[Res]:
it = _iter_favs()
# trick to sort errors along with the actual objects
# TODO wonder if there is a shorter way?
# TODO add to the error handling post?
favs = list(it)
prev = datetime.min
keys = []
for i, f in enumerate(favs):
if not isinstance(f, Exception):
prev = f.dt
keys.append((prev, i)) # include index to resolve ties
sorted_items = [p[1] for p in sorted(zip(keys, favs))]
#
return sorted_items
def stats() -> Stats:
return stat(favorites)