2020-12-07 18:32:31 +00:00
|
|
|
''' tests the base functionality for activitypub dataclasses '''
|
2020-12-07 20:19:15 +00:00
|
|
|
from io import BytesIO
|
2020-12-07 18:46:41 +00:00
|
|
|
import json
|
|
|
|
import pathlib
|
|
|
|
from unittest.mock import patch
|
|
|
|
|
2020-12-07 18:32:31 +00:00
|
|
|
from dataclasses import dataclass
|
|
|
|
from django.test import TestCase
|
2020-12-07 20:19:15 +00:00
|
|
|
from PIL import Image
|
2020-12-07 18:46:41 +00:00
|
|
|
import responses
|
2020-12-07 18:32:31 +00:00
|
|
|
|
2020-12-07 20:19:15 +00:00
|
|
|
from bookwyrm import activitypub
|
2020-12-07 18:32:31 +00:00
|
|
|
from bookwyrm.activitypub.base_activity import ActivityObject, \
|
2020-12-12 21:39:55 +00:00
|
|
|
resolve_remote_id, set_related_field
|
2020-12-07 18:32:31 +00:00
|
|
|
from bookwyrm.activitypub import ActivitySerializerError
|
|
|
|
from bookwyrm import models
|
|
|
|
|
|
|
|
class BaseActivity(TestCase):
|
|
|
|
''' the super class for model-linked activitypub dataclasses '''
|
2020-12-07 20:19:15 +00:00
|
|
|
def setUp(self):
|
|
|
|
''' we're probably going to re-use this so why copy/paste '''
|
|
|
|
self.user = models.User.objects.create_user(
|
2020-12-27 23:18:48 +00:00
|
|
|
'mouse', 'mouse@mouse.mouse', 'mouseword',
|
|
|
|
local=True, localname='mouse')
|
2020-12-07 20:19:15 +00:00
|
|
|
self.user.remote_id = 'http://example.com/a/b'
|
2021-02-08 17:38:28 +00:00
|
|
|
self.user.save(broadcast=False)
|
2020-12-07 20:19:15 +00:00
|
|
|
|
|
|
|
datafile = pathlib.Path(__file__).parent.joinpath(
|
|
|
|
'../data/ap_user.json'
|
|
|
|
)
|
|
|
|
self.userdata = json.loads(datafile.read_bytes())
|
|
|
|
# don't try to load the user icon
|
|
|
|
del self.userdata['icon']
|
|
|
|
|
2020-12-08 02:28:42 +00:00
|
|
|
image_file = pathlib.Path(__file__).parent.joinpath(
|
|
|
|
'../../static/images/default_avi.jpg')
|
|
|
|
image = Image.open(image_file)
|
|
|
|
output = BytesIO()
|
|
|
|
image.save(output, format=image.format)
|
|
|
|
self.image_data = output.getvalue()
|
|
|
|
|
2020-12-07 18:32:31 +00:00
|
|
|
def test_init(self):
|
|
|
|
''' simple successfuly init '''
|
|
|
|
instance = ActivityObject(id='a', type='b')
|
|
|
|
self.assertTrue(hasattr(instance, 'id'))
|
|
|
|
self.assertTrue(hasattr(instance, 'type'))
|
|
|
|
|
|
|
|
def test_init_missing(self):
|
|
|
|
''' init with missing required params '''
|
|
|
|
with self.assertRaises(ActivitySerializerError):
|
|
|
|
ActivityObject()
|
|
|
|
|
|
|
|
def test_init_extra_fields(self):
|
|
|
|
''' init ignoring additional fields '''
|
|
|
|
instance = ActivityObject(id='a', type='b', fish='c')
|
|
|
|
self.assertTrue(hasattr(instance, 'id'))
|
|
|
|
self.assertTrue(hasattr(instance, 'type'))
|
|
|
|
|
|
|
|
def test_init_default_field(self):
|
|
|
|
''' replace an existing required field with a default field '''
|
|
|
|
@dataclass(init=False)
|
|
|
|
class TestClass(ActivityObject):
|
|
|
|
''' test class with default field '''
|
|
|
|
type: str = 'TestObject'
|
|
|
|
|
|
|
|
instance = TestClass(id='a')
|
|
|
|
self.assertEqual(instance.id, 'a')
|
|
|
|
self.assertEqual(instance.type, 'TestObject')
|
|
|
|
|
|
|
|
def test_serialize(self):
|
|
|
|
''' simple function for converting dataclass to dict '''
|
|
|
|
instance = ActivityObject(id='a', type='b')
|
|
|
|
serialized = instance.serialize()
|
|
|
|
self.assertIsInstance(serialized, dict)
|
|
|
|
self.assertEqual(serialized['id'], 'a')
|
|
|
|
self.assertEqual(serialized['type'], 'b')
|
|
|
|
|
2020-12-07 18:46:41 +00:00
|
|
|
@responses.activate
|
|
|
|
def test_resolve_remote_id(self):
|
|
|
|
''' look up or load remote data '''
|
|
|
|
# existing item
|
2021-02-17 03:35:43 +00:00
|
|
|
result = resolve_remote_id('http://example.com/a/b', model=models.User)
|
2020-12-07 20:19:15 +00:00
|
|
|
self.assertEqual(result, self.user)
|
2020-12-07 18:46:41 +00:00
|
|
|
|
|
|
|
# remote item
|
|
|
|
responses.add(
|
|
|
|
responses.GET,
|
|
|
|
'https://example.com/user/mouse',
|
2020-12-07 20:19:15 +00:00
|
|
|
json=self.userdata,
|
2020-12-07 18:46:41 +00:00
|
|
|
status=200)
|
|
|
|
|
|
|
|
with patch('bookwyrm.models.user.set_remote_server.delay'):
|
|
|
|
result = resolve_remote_id(
|
2021-02-17 03:35:43 +00:00
|
|
|
'https://example.com/user/mouse', model=models.User)
|
2020-12-07 18:46:41 +00:00
|
|
|
self.assertIsInstance(result, models.User)
|
|
|
|
self.assertEqual(result.remote_id, 'https://example.com/user/mouse')
|
|
|
|
self.assertEqual(result.name, 'MOUSE?? MOUSE!!')
|
2020-12-07 20:19:15 +00:00
|
|
|
|
2020-12-13 22:35:56 +00:00
|
|
|
def test_to_model_invalid_model(self):
|
|
|
|
''' catch mismatch between activity type and model type '''
|
2020-12-07 20:19:15 +00:00
|
|
|
instance = ActivityObject(id='a', type='b')
|
|
|
|
with self.assertRaises(ActivitySerializerError):
|
2021-02-17 04:17:38 +00:00
|
|
|
instance.to_model(model=models.User)
|
2020-12-07 20:19:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
@responses.activate
|
|
|
|
def test_to_model_image(self):
|
|
|
|
''' update an image field '''
|
2020-12-13 22:35:56 +00:00
|
|
|
activity = activitypub.Person(
|
|
|
|
id=self.user.remote_id,
|
|
|
|
name='New Name',
|
|
|
|
preferredUsername='mouse',
|
|
|
|
inbox='http://www.com/',
|
|
|
|
outbox='http://www.com/',
|
|
|
|
followers='',
|
|
|
|
summary='',
|
2021-02-17 04:17:38 +00:00
|
|
|
publicKey={
|
|
|
|
'id': 'hi',
|
|
|
|
'owner': self.user.remote_id,
|
|
|
|
'publicKeyPem': 'hi'},
|
2020-12-13 22:35:56 +00:00
|
|
|
endpoints={},
|
2021-02-17 04:17:38 +00:00
|
|
|
icon={
|
|
|
|
'type': 'Image',
|
|
|
|
'url': 'http://www.example.com/image.jpg'
|
|
|
|
}
|
2020-12-13 22:35:56 +00:00
|
|
|
)
|
|
|
|
|
2020-12-07 20:19:15 +00:00
|
|
|
responses.add(
|
|
|
|
responses.GET,
|
|
|
|
'http://www.example.com/image.jpg',
|
2020-12-08 02:28:42 +00:00
|
|
|
body=self.image_data,
|
2020-12-07 20:19:15 +00:00
|
|
|
status=200)
|
|
|
|
|
|
|
|
self.assertIsNone(self.user.avatar.name)
|
|
|
|
with self.assertRaises(ValueError):
|
|
|
|
self.user.avatar.file #pylint: disable=pointless-statement
|
|
|
|
|
2021-02-08 17:38:28 +00:00
|
|
|
# this would trigger a broadcast because it's a local user
|
|
|
|
with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
|
2021-02-17 04:17:38 +00:00
|
|
|
activity.to_model(model=models.User, instance=self.user)
|
2020-12-07 20:19:15 +00:00
|
|
|
self.assertIsNotNone(self.user.avatar.file)
|
2021-02-17 04:24:37 +00:00
|
|
|
self.assertEqual(self.user.name, 'New Name')
|
|
|
|
self.assertEqual(self.user.key_pair.public_key, 'hi')
|
2020-12-07 21:16:42 +00:00
|
|
|
|
|
|
|
def test_to_model_many_to_many(self):
|
|
|
|
''' annoying that these all need special handling '''
|
2021-02-08 17:38:28 +00:00
|
|
|
with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
|
|
|
|
status = models.Status.objects.create(
|
|
|
|
content='test status',
|
|
|
|
user=self.user,
|
|
|
|
)
|
2020-12-12 22:15:10 +00:00
|
|
|
book = models.Edition.objects.create(
|
|
|
|
title='Test Edition', remote_id='http://book.com/book')
|
2020-12-13 22:35:56 +00:00
|
|
|
update_data = activitypub.Note(
|
|
|
|
id=status.remote_id,
|
|
|
|
content=status.content,
|
|
|
|
attributedTo=self.user.remote_id,
|
|
|
|
published='hi',
|
|
|
|
to=[],
|
|
|
|
cc=[],
|
|
|
|
tag=[
|
|
|
|
{
|
|
|
|
'type': 'Mention',
|
|
|
|
'name': 'gerald',
|
|
|
|
'href': 'http://example.com/a/b'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
'type': 'Edition',
|
|
|
|
'name': 'gerald j. books',
|
|
|
|
'href': 'http://book.com/book'
|
|
|
|
},
|
|
|
|
]
|
|
|
|
)
|
2021-02-17 04:17:38 +00:00
|
|
|
update_data.to_model(model=models.Status, instance=status)
|
2020-12-07 21:16:42 +00:00
|
|
|
self.assertEqual(status.mention_users.first(), self.user)
|
2020-12-12 22:15:10 +00:00
|
|
|
self.assertEqual(status.mention_books.first(), book)
|
2020-12-08 02:28:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
@responses.activate
|
|
|
|
def test_to_model_one_to_many(self):
|
|
|
|
''' these are reversed relationships, where the secondary object
|
|
|
|
keys the primary object but not vice versa '''
|
2021-02-08 17:38:28 +00:00
|
|
|
with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
|
|
|
|
status = models.Status.objects.create(
|
|
|
|
content='test status',
|
|
|
|
user=self.user,
|
|
|
|
)
|
2020-12-13 22:35:56 +00:00
|
|
|
update_data = activitypub.Note(
|
|
|
|
id=status.remote_id,
|
|
|
|
content=status.content,
|
|
|
|
attributedTo=self.user.remote_id,
|
|
|
|
published='hi',
|
|
|
|
to=[],
|
|
|
|
cc=[],
|
|
|
|
attachment=[{
|
|
|
|
'url': 'http://www.example.com/image.jpg',
|
|
|
|
'name': 'alt text',
|
|
|
|
'type': 'Image',
|
|
|
|
}],
|
|
|
|
)
|
2020-12-08 02:28:42 +00:00
|
|
|
|
|
|
|
responses.add(
|
|
|
|
responses.GET,
|
|
|
|
'http://www.example.com/image.jpg',
|
|
|
|
body=self.image_data,
|
|
|
|
status=200)
|
|
|
|
|
2020-12-09 21:11:42 +00:00
|
|
|
# sets the celery task call to the function call
|
|
|
|
with patch(
|
|
|
|
'bookwyrm.activitypub.base_activity.set_related_field.delay'):
|
2021-02-17 04:17:38 +00:00
|
|
|
update_data.to_model(model=models.Status, instance=status)
|
2020-12-09 21:11:42 +00:00
|
|
|
self.assertIsNone(status.attachments.first())
|
|
|
|
|
|
|
|
|
|
|
|
@responses.activate
|
|
|
|
def test_set_related_field(self):
|
|
|
|
''' celery task to add back-references to created objects '''
|
2021-02-08 17:38:28 +00:00
|
|
|
with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
|
|
|
|
status = models.Status.objects.create(
|
|
|
|
content='test status',
|
|
|
|
user=self.user,
|
|
|
|
)
|
2020-12-09 21:11:42 +00:00
|
|
|
data = {
|
|
|
|
'url': 'http://www.example.com/image.jpg',
|
|
|
|
'name': 'alt text',
|
|
|
|
'type': 'Image',
|
|
|
|
}
|
|
|
|
responses.add(
|
|
|
|
responses.GET,
|
|
|
|
'http://www.example.com/image.jpg',
|
|
|
|
body=self.image_data,
|
|
|
|
status=200)
|
|
|
|
set_related_field(
|
|
|
|
'Image', 'Status', 'status', status.remote_id, data)
|
|
|
|
|
2020-12-08 02:28:42 +00:00
|
|
|
self.assertIsInstance(status.attachments.first(), models.Image)
|
2020-12-09 21:11:42 +00:00
|
|
|
self.assertIsNotNone(status.attachments.first().image)
|