2021-04-08 18:18:32 +00:00
|
|
|
""" tests incoming activities"""
|
2021-04-22 15:31:32 +00:00
|
|
|
import json
|
2021-04-08 18:18:32 +00:00
|
|
|
from unittest.mock import patch
|
|
|
|
|
|
|
|
from django.test import TestCase
|
|
|
|
|
|
|
|
from bookwyrm import models, views
|
|
|
|
|
|
|
|
|
|
|
|
# pylint: disable=too-many-public-methods
|
|
|
|
class InboxRelationships(TestCase):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""inbox tests"""
|
2021-04-08 18:18:32 +00:00
|
|
|
|
|
|
|
def setUp(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""basic user and book data"""
|
2021-09-06 21:48:45 +00:00
|
|
|
with patch("bookwyrm.suggested_users.rerank_suggestions_task.delay"), patch(
|
|
|
|
"bookwyrm.activitystreams.populate_stream_task.delay"
|
2021-12-09 23:02:53 +00:00
|
|
|
), patch("bookwyrm.lists_stream.populate_lists_task.delay"):
|
2021-08-03 20:27:32 +00:00
|
|
|
self.local_user = models.User.objects.create_user(
|
|
|
|
"mouse@example.com",
|
|
|
|
"mouse@mouse.com",
|
|
|
|
"mouseword",
|
|
|
|
local=True,
|
|
|
|
localname="mouse",
|
|
|
|
)
|
2021-08-02 23:05:40 +00:00
|
|
|
self.local_user.remote_id = "https://example.com/user/mouse"
|
2021-08-03 20:27:32 +00:00
|
|
|
self.local_user.save(broadcast=False, update_fields=["remote_id"])
|
2021-08-02 23:05:40 +00:00
|
|
|
with patch("bookwyrm.models.user.set_remote_server.delay"):
|
|
|
|
self.remote_user = models.User.objects.create_user(
|
|
|
|
"rat",
|
|
|
|
"rat@rat.com",
|
|
|
|
"ratword",
|
|
|
|
local=False,
|
|
|
|
remote_id="https://example.com/users/rat",
|
|
|
|
inbox="https://example.com/users/rat/inbox",
|
|
|
|
outbox="https://example.com/users/rat/outbox",
|
2021-05-26 21:57:29 +00:00
|
|
|
)
|
|
|
|
|
2021-08-02 23:05:40 +00:00
|
|
|
models.SiteSettings.objects.create()
|
2021-04-08 18:18:32 +00:00
|
|
|
|
2021-04-22 15:31:32 +00:00
|
|
|
def test_follow(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""remote user wants to follow local user"""
|
2021-04-08 18:18:32 +00:00
|
|
|
activity = {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": "https://example.com/users/rat/follows/123",
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": "https://example.com/user/mouse",
|
|
|
|
}
|
|
|
|
|
|
|
|
self.assertFalse(models.UserFollowRequest.objects.exists())
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch(
|
|
|
|
"bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"
|
|
|
|
) as mock:
|
2021-04-08 18:18:32 +00:00
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
self.assertEqual(mock.call_count, 1)
|
2021-11-12 17:17:00 +00:00
|
|
|
response_activity = json.loads(mock.call_args[1]["args"][1])
|
2021-04-22 15:31:32 +00:00
|
|
|
self.assertEqual(response_activity["type"], "Accept")
|
2021-04-08 18:18:32 +00:00
|
|
|
|
|
|
|
# notification created
|
|
|
|
notification = models.Notification.objects.get()
|
|
|
|
self.assertEqual(notification.user, self.local_user)
|
|
|
|
self.assertEqual(notification.notification_type, "FOLLOW")
|
|
|
|
|
|
|
|
# the request should have been deleted
|
|
|
|
self.assertFalse(models.UserFollowRequest.objects.exists())
|
|
|
|
|
|
|
|
# the follow relationship should exist
|
|
|
|
follow = models.UserFollows.objects.get(user_object=self.local_user)
|
|
|
|
self.assertEqual(follow.user_subject, self.remote_user)
|
|
|
|
|
2021-04-22 15:31:32 +00:00
|
|
|
def test_follow_duplicate(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""remote user wants to follow local user twice"""
|
2021-04-22 15:31:32 +00:00
|
|
|
activity = {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": "https://example.com/users/rat/follows/123",
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": "https://example.com/user/mouse",
|
|
|
|
}
|
|
|
|
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-22 15:31:32 +00:00
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
|
|
|
|
# the follow relationship should exist
|
|
|
|
follow = models.UserFollows.objects.get(user_object=self.local_user)
|
|
|
|
self.assertEqual(follow.user_subject, self.remote_user)
|
|
|
|
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch(
|
|
|
|
"bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"
|
|
|
|
) as mock:
|
2021-04-22 15:31:32 +00:00
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
self.assertEqual(mock.call_count, 1)
|
2021-11-12 17:17:00 +00:00
|
|
|
response_activity = json.loads(mock.call_args[1]["args"][1])
|
2021-04-22 15:31:32 +00:00
|
|
|
self.assertEqual(response_activity["type"], "Accept")
|
|
|
|
|
|
|
|
# the follow relationship should STILL exist
|
|
|
|
follow = models.UserFollows.objects.get(user_object=self.local_user)
|
|
|
|
self.assertEqual(follow.user_subject, self.remote_user)
|
|
|
|
|
|
|
|
def test_follow_manually_approved(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""needs approval before following"""
|
2021-04-08 18:18:32 +00:00
|
|
|
activity = {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": "https://example.com/users/rat/follows/123",
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": "https://example.com/user/mouse",
|
|
|
|
}
|
|
|
|
|
|
|
|
self.local_user.manually_approves_followers = True
|
2021-08-03 21:05:52 +00:00
|
|
|
self.local_user.save(
|
|
|
|
broadcast=False, update_fields=["manually_approves_followers"]
|
|
|
|
)
|
2021-04-08 18:18:32 +00:00
|
|
|
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-08 18:18:32 +00:00
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
|
|
|
|
# notification created
|
|
|
|
notification = models.Notification.objects.get()
|
|
|
|
self.assertEqual(notification.user, self.local_user)
|
|
|
|
self.assertEqual(notification.notification_type, "FOLLOW_REQUEST")
|
|
|
|
|
|
|
|
# the request should exist
|
|
|
|
request = models.UserFollowRequest.objects.get()
|
|
|
|
self.assertEqual(request.user_subject, self.remote_user)
|
|
|
|
self.assertEqual(request.user_object, self.local_user)
|
|
|
|
|
|
|
|
# the follow relationship should not exist
|
|
|
|
follow = models.UserFollows.objects.all()
|
|
|
|
self.assertEqual(list(follow), [])
|
|
|
|
|
2021-04-22 15:31:32 +00:00
|
|
|
def test_undo_follow_request(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""the requester cancels a follow request"""
|
2021-04-08 18:18:32 +00:00
|
|
|
self.local_user.manually_approves_followers = True
|
2021-08-03 21:05:52 +00:00
|
|
|
self.local_user.save(
|
|
|
|
broadcast=False, update_fields=["manually_approves_followers"]
|
|
|
|
)
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-08 18:18:32 +00:00
|
|
|
request = models.UserFollowRequest.objects.create(
|
|
|
|
user_subject=self.remote_user, user_object=self.local_user
|
|
|
|
)
|
|
|
|
self.assertTrue(self.local_user.follower_requests.exists())
|
|
|
|
|
|
|
|
activity = {
|
|
|
|
"type": "Undo",
|
|
|
|
"id": "bleh",
|
|
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
|
|
"cc": ["https://example.com/user/mouse/followers"],
|
|
|
|
"actor": self.remote_user.remote_id,
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"object": {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": request.remote_id,
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": "https://example.com/user/mouse",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
|
|
|
|
self.assertFalse(self.local_user.follower_requests.exists())
|
|
|
|
|
2021-04-22 15:31:32 +00:00
|
|
|
def test_unfollow(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""remove a relationship"""
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-08 18:18:32 +00:00
|
|
|
rel = models.UserFollows.objects.create(
|
|
|
|
user_subject=self.remote_user, user_object=self.local_user
|
|
|
|
)
|
|
|
|
activity = {
|
|
|
|
"type": "Undo",
|
|
|
|
"id": "bleh",
|
|
|
|
"to": ["https://www.w3.org/ns/activitystreams#Public"],
|
|
|
|
"cc": ["https://example.com/user/mouse/followers"],
|
|
|
|
"actor": self.remote_user.remote_id,
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"object": {
|
|
|
|
"id": rel.remote_id,
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": "https://example.com/user/mouse",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
self.assertEqual(self.remote_user, self.local_user.followers.first())
|
|
|
|
|
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
self.assertIsNone(self.local_user.followers.first())
|
|
|
|
|
2021-09-07 01:39:14 +00:00
|
|
|
@patch("bookwyrm.activitystreams.add_user_statuses_task.delay")
|
2021-12-09 23:02:53 +00:00
|
|
|
@patch("bookwyrm.lists_stream.add_user_lists_task.delay")
|
|
|
|
def test_follow_accept(self, *_):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""a remote user approved a follow request from local"""
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-08 18:18:32 +00:00
|
|
|
rel = models.UserFollowRequest.objects.create(
|
|
|
|
user_subject=self.local_user, user_object=self.remote_user
|
|
|
|
)
|
|
|
|
activity = {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": "https://example.com/users/rat/follows/123#accepts",
|
|
|
|
"type": "Accept",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": {
|
|
|
|
"id": rel.remote_id,
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/user/mouse",
|
|
|
|
"object": "https://example.com/users/rat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
self.assertEqual(models.UserFollowRequest.objects.count(), 1)
|
|
|
|
|
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
|
|
|
|
# request should be deleted
|
|
|
|
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
|
|
|
|
|
|
|
|
# relationship should be created
|
|
|
|
follows = self.remote_user.followers
|
|
|
|
self.assertEqual(follows.count(), 1)
|
|
|
|
self.assertEqual(follows.first(), self.local_user)
|
|
|
|
|
2021-04-22 15:31:32 +00:00
|
|
|
def test_follow_reject(self):
|
2021-04-26 16:15:42 +00:00
|
|
|
"""turn down a follow request"""
|
2021-11-12 17:17:00 +00:00
|
|
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.apply_async"):
|
2021-04-08 18:18:32 +00:00
|
|
|
rel = models.UserFollowRequest.objects.create(
|
|
|
|
user_subject=self.local_user, user_object=self.remote_user
|
|
|
|
)
|
|
|
|
activity = {
|
|
|
|
"@context": "https://www.w3.org/ns/activitystreams",
|
|
|
|
"id": "https://example.com/users/rat/follows/123#accepts",
|
|
|
|
"type": "Reject",
|
|
|
|
"actor": "https://example.com/users/rat",
|
|
|
|
"object": {
|
|
|
|
"id": rel.remote_id,
|
|
|
|
"type": "Follow",
|
|
|
|
"actor": "https://example.com/user/mouse",
|
|
|
|
"object": "https://example.com/users/rat",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
self.assertEqual(models.UserFollowRequest.objects.count(), 1)
|
|
|
|
|
|
|
|
views.inbox.activity_task(activity)
|
|
|
|
|
|
|
|
# request should be deleted
|
|
|
|
self.assertFalse(models.UserFollowRequest.objects.exists())
|
|
|
|
self.assertFalse(self.remote_user.followers.exists())
|