mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-02 05:18:43 +00:00
8bbf1fe252
* Use dataclasses to define activitypub (de)serialization
129 lines
3.4 KiB
Python
129 lines
3.4 KiB
Python
''' manage remote users '''
|
|
from urllib.parse import urlparse
|
|
from uuid import uuid4
|
|
import requests
|
|
|
|
from django.core.files.base import ContentFile
|
|
from django.db import transaction
|
|
|
|
from fedireads import activitypub, models
|
|
|
|
|
|
def get_or_create_remote_user(actor):
|
|
''' look up a remote user or add them '''
|
|
try:
|
|
return models.User.objects.get(remote_id=actor)
|
|
except models.User.DoesNotExist:
|
|
pass
|
|
|
|
data = fetch_user_data(actor)
|
|
|
|
actor_parts = urlparse(actor)
|
|
with transaction.atomic():
|
|
user = create_remote_user(data)
|
|
user.federated_server = get_or_create_remote_server(actor_parts.netloc)
|
|
user.save()
|
|
|
|
avatar = get_avatar(data)
|
|
if avatar:
|
|
user.avatar.save(*avatar)
|
|
|
|
if user.fedireads_user:
|
|
get_remote_reviews(user)
|
|
return user
|
|
|
|
|
|
def fetch_user_data(actor):
|
|
''' load the user's info from the actor url '''
|
|
response = requests.get(
|
|
actor,
|
|
headers={'Accept': 'application/activity+json'}
|
|
)
|
|
if not response.ok:
|
|
response.raise_for_status()
|
|
data = response.json()
|
|
|
|
# make sure our actor is who they say they are
|
|
if actor != data['id']:
|
|
raise ValueError("Remote actor id must match url.")
|
|
return data
|
|
|
|
|
|
def create_remote_user(data):
|
|
''' parse the activitypub actor data into a user '''
|
|
actor = activitypub.Person(**data)
|
|
return actor.to_model(models.User)
|
|
|
|
|
|
def refresh_remote_user(user):
|
|
''' get updated user data from its home instance '''
|
|
data = fetch_user_data(user.remote_id)
|
|
|
|
activity = activitypub.Person(**data)
|
|
activity.to_model(models.User, instance=user)
|
|
|
|
|
|
def get_avatar(data):
|
|
''' find the icon attachment and load the image from the remote sever '''
|
|
icon_blob = data.get('icon')
|
|
if not icon_blob or not icon_blob.get('url'):
|
|
return None
|
|
|
|
response = requests.get(icon_blob['url'])
|
|
if not response.ok:
|
|
return None
|
|
|
|
image_name = str(uuid4()) + '.' + icon_blob['url'].split('.')[-1]
|
|
image_content = ContentFile(response.content)
|
|
return [image_name, image_content]
|
|
|
|
|
|
def get_remote_reviews(user):
|
|
''' ingest reviews by a new remote fedireads user '''
|
|
outbox_page = user.outbox + '?page=true'
|
|
response = requests.get(
|
|
outbox_page,
|
|
headers={'Accept': 'application/activity+json'}
|
|
)
|
|
data = response.json()
|
|
# TODO: pagination?
|
|
for status in data['orderedItems']:
|
|
if status.get('fedireadsType') == 'Review':
|
|
activitypub.Review(**status).to_model(models.Review)
|
|
|
|
|
|
def get_or_create_remote_server(domain):
|
|
''' get info on a remote server '''
|
|
try:
|
|
return models.FederatedServer.objects.get(
|
|
server_name=domain
|
|
)
|
|
except models.FederatedServer.DoesNotExist:
|
|
pass
|
|
|
|
response = requests.get(
|
|
'https://%s/.well-known/nodeinfo' % domain,
|
|
headers={'Accept': 'application/activity+json'}
|
|
)
|
|
|
|
if response.status_code != 200:
|
|
return None
|
|
|
|
data = response.json()
|
|
try:
|
|
nodeinfo_url = data.get('links')[0].get('href')
|
|
except (TypeError, KeyError):
|
|
return None
|
|
|
|
response = requests.get(
|
|
nodeinfo_url,
|
|
headers={'Accept': 'application/activity+json'}
|
|
)
|
|
data = response.json()
|
|
|
|
server = models.FederatedServer.objects.create(
|
|
server_name=domain,
|
|
application_type=data['software']['name'],
|
|
application_version=data['software']['version'],
|
|
)
|
|
return server
|