moviewyrm/bookwyrm/remote_user.py

140 lines
3.6 KiB
Python
Raw Normal View History

2020-02-11 23:17:21 +00:00
''' manage remote users '''
from urllib.parse import urlparse
2020-04-03 17:19:04 +00:00
from uuid import uuid4
2020-04-22 13:53:22 +00:00
import requests
2020-04-03 17:19:04 +00:00
from django.core.files.base import ContentFile
2020-05-14 19:08:57 +00:00
from django.db import transaction
2020-02-11 23:17:21 +00:00
from bookwyrm import activitypub, models
from bookwyrm import status as status_builder
from bookwyrm.tasks import app
2020-02-11 23:17:21 +00:00
def get_or_create_remote_user(actor):
''' look up a remote user or add them '''
try:
2020-05-14 19:08:57 +00:00
return models.User.objects.get(remote_id=actor)
2020-02-11 23:17:21 +00:00
except models.User.DoesNotExist:
pass
data = fetch_user_data(actor)
2020-02-11 23:17:21 +00:00
actor_parts = urlparse(actor)
2020-05-14 19:08:57 +00:00
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)
2020-05-14 19:08:57 +00:00
if user.bookwyrm_user:
get_remote_reviews.delay(user.id)
2020-05-14 19:08:57 +00:00
return user
def fetch_user_data(actor):
''' load the user's info from the actor url '''
2020-11-12 20:27:49 +00:00
try:
response = requests.get(
actor,
headers={'Accept': 'application/activity+json'}
)
except ConnectionError:
return None
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
2020-05-14 19:08:57 +00:00
def create_remote_user(data):
''' parse the activitypub actor data into a user '''
actor = activitypub.Person(**data)
return actor.to_model(models.User)
2020-05-14 19:08:57 +00:00
2020-02-11 23:17:21 +00:00
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)
2020-02-11 23:17:21 +00:00
2020-04-03 17:19:04 +00:00
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]
@app.task
def get_remote_reviews(user_id):
''' ingest reviews by a new remote bookwyrm user '''
2020-11-12 20:27:49 +00:00
try:
user = models.User.objects.get(id=user_id)
except models.User.DoesNotExist:
return
outbox_page = user.outbox + '?page=true'
response = requests.get(
outbox_page,
headers={'Accept': 'application/activity+json'}
)
data = response.json()
# TODO: pagination?
for activity in data['orderedItems']:
status_builder.create_status(activity)
2020-03-29 22:51:43 +00:00
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
2020-03-29 22:51:43 +00:00
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