lib.itmens/takahe/ap_handlers.py
2024-06-15 22:02:19 -04:00

209 lines
6.2 KiB
Python

from time import sleep
from typing import Any
from django.conf import settings
from loguru import logger
from catalog.common import *
from journal.models import (
Comment,
Content,
Note,
Piece,
PieceInteraction,
Rating,
Review,
ShelfMember,
)
from users.middlewares import activate_language_for_user
from users.models.apidentity import APIdentity
from .models import Follow, Identity, Post, TimelineEvent
from .utils import Takahe
_supported_ap_catalog_item_types = [
"Edition",
"Movie",
"TVShow",
"TVSeason",
"TVEpisode",
"Album",
"Game",
"Podcast",
"PodcastEpisode",
"Performance",
"PerformanceProduction",
]
_supported_ap_journal_types = {
"Status": ShelfMember,
"Rating": Rating,
"Comment": Comment,
"Review": Review,
"Note": Note,
}
def _parse_items(objects) -> list[dict[str, Any]]:
logger.debug(f"Parsing item links from {objects}")
if not objects:
return []
objs = objects if isinstance(objects, list) else [objects]
items = [obj for obj in objs if obj["type"] in _supported_ap_catalog_item_types]
return items
def _parse_piece_objects(objects) -> list[dict[str, Any]]:
logger.debug(f"Parsing pieces from {objects}")
if not objects:
return []
objs = objects if isinstance(objects, list) else [objects]
pieces = []
for obj in objs:
if obj["type"] in _supported_ap_journal_types.keys():
pieces.append(obj)
else:
logger.warning(f'Unknown link type {obj["type"]}')
return pieces
def _get_or_create_item(item_obj) -> Item | None:
logger.debug(f"Fetching item by ap from {item_obj}")
typ = item_obj["type"]
url = item_obj["href"]
if url.startswith(settings.SITE_INFO["site_url"]):
item = Item.get_by_url(url, True)
if not item:
logger.warning(f"Item not found for {url}")
return item
if typ in ["TVEpisode", "PodcastEpisode"]:
# TODO support episode item
# match and fetch parent item first
logger.debug(f"{typ}:{url} not supported yet")
return None
site = SiteManager.get_site_by_url(url)
if not site:
logger.warning(f"Site not found for {url}")
return None
site.get_resource_ready()
item = site.get_item()
if not item:
logger.warning(f"Item not fetched for {url}")
return item
def post_created(pk, post_data):
return post_fetched(pk, post_data)
def post_edited(pk, post_data):
return post_fetched(pk, post_data)
def post_fetched(pk, post_data):
post = Post.objects.get(pk=pk)
owner = Takahe.get_or_create_remote_apidentity(post.author)
activate_language_for_user(owner.user)
if not post.type_data and not post_data:
logger.warning(f"Post {post} has no type_data")
return
ap_objects = post_data or post.type_data.get("object", {})
items = _parse_items(ap_objects.get("tag"))
pieces = _parse_piece_objects(ap_objects.get("relatedWith"))
logger.info(f"Post {post} has items {items} and pieces {pieces}")
if len(items) == 0:
logger.warning(f"Post {post} has no remote items")
return
elif len(items) > 1:
logger.warning(f"Post {post} has more than one remote item")
return
item = _get_or_create_item(items[0])
if not item:
logger.warning(f"Post {post} has no local item matched or created")
return
for p in pieces:
cls = _supported_ap_journal_types.get(p["type"])
if not cls:
logger.warning(f'Unknown link type {p["type"]}')
continue
cls.update_by_ap_object(owner, item, p, post)
def post_deleted(pk, post_data):
for piece in Piece.objects.filter(posts__id=pk):
if piece.local and piece.__class__ != Note:
# no delete other than Note, for backward compatibility, should reconsider later
return
# delete piece if the deleted post is the most recent one for the piece
if piece.latest_post_id == pk:
logger.debug(f"Deleting piece {piece}")
piece.delete()
else:
logger.debug(f"Matched piece {piece} has newer posts, not deleting")
def post_interacted(interaction_pk, interaction, post_pk, identity_pk):
if interaction not in ["like", "boost", "pin"]:
return
p = Piece.objects.filter(posts__id=post_pk).first()
if not p:
return
apid = APIdentity.objects.filter(pk=identity_pk).first()
if not apid:
logger.warning(f"Identity {identity_pk} not found for interaction")
return
if (
interaction == "boost"
and p.local
and p.owner.user.mastodon_acct == apid.full_handle
):
# ignore boost by oneself
TimelineEvent.objects.filter(
identity_id=p.owner_id,
type="boosted",
subject_post_id=post_pk,
subject_identity_id=identity_pk,
).delete()
return
PieceInteraction.objects.get_or_create(
target=p,
identity_id=identity_pk,
interaction_type=interaction,
defaults={"target_type": p.__class__.__name__},
)
def post_uninteracted(interaction_pk, interaction, post_pk, identity_pk):
if interaction not in ["like", "boost", "pin"]:
return
p = Piece.objects.filter(posts__id=post_pk).first()
if not p:
return
if not APIdentity.objects.filter(pk=identity_pk).exists():
logger.warning(f"Identity {identity_pk} not found for interaction")
return
PieceInteraction.objects.filter(
target=p,
identity_id=identity_pk,
interaction_type=interaction,
).delete()
def identity_fetched(pk):
try:
identity = Identity.objects.get(pk=pk)
except Identity.DoesNotExist:
sleep(2)
try:
identity = Identity.objects.get(pk=pk)
except Identity.DoesNotExist:
logger.warning(f"Fetched identity {pk} not found")
return
if identity.username and identity.domain:
apid = Takahe.get_or_create_remote_apidentity(identity)
if apid:
logger.debug(f"Identity {identity} synced")
else:
logger.warning(f"Identity {identity} not synced")
else:
logger.warning(f"Identity {identity} has no username or domain")