forked from mirrors/bookwyrm
Merge branch 'main' into book-data-model
This commit is contained in:
commit
77948f64d2
10 changed files with 317 additions and 232 deletions
|
@ -3,7 +3,7 @@ from dataclasses import dataclass, fields, MISSING
|
||||||
from json import JSONEncoder
|
from json import JSONEncoder
|
||||||
|
|
||||||
from django.apps import apps
|
from django.apps import apps
|
||||||
from django.db import transaction
|
from django.db import IntegrityError, transaction
|
||||||
|
|
||||||
from bookwyrm.connectors import ConnectorException, get_data
|
from bookwyrm.connectors import ConnectorException, get_data
|
||||||
from bookwyrm.tasks import app
|
from bookwyrm.tasks import app
|
||||||
|
@ -92,7 +92,10 @@ class ActivityObject:
|
||||||
|
|
||||||
with transaction.atomic():
|
with transaction.atomic():
|
||||||
# we can't set many to many and reverse fields on an unsaved object
|
# we can't set many to many and reverse fields on an unsaved object
|
||||||
|
try:
|
||||||
instance.save()
|
instance.save()
|
||||||
|
except IntegrityError as e:
|
||||||
|
raise ActivitySerializerError(e)
|
||||||
|
|
||||||
# add many to many fields, which have to be set post-save
|
# add many to many fields, which have to be set post-save
|
||||||
for field in instance.many_to_many_fields:
|
for field in instance.many_to_many_fields:
|
||||||
|
|
52
bookwyrm/tests/data/ap_generated_shelve_note.json
Normal file
52
bookwyrm/tests/data/ap_generated_shelve_note.json
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
{
|
||||||
|
"id": "https://example.com/users/rat/generatednote/2567/activity",
|
||||||
|
"type": "Create",
|
||||||
|
"actor": "https://example.com/users/rat",
|
||||||
|
"object": {
|
||||||
|
"id": "https://example.com/users/rat/generatednote/2567",
|
||||||
|
"type": "GeneratedNote",
|
||||||
|
"url": null,
|
||||||
|
"inReplyTo": null,
|
||||||
|
"published": "2020-12-16T01:45:19.662734+00:00",
|
||||||
|
"attributedTo": "https://example.com/users/rat",
|
||||||
|
"to": [
|
||||||
|
"https://www.w3.org/ns/activitystreams#Public"
|
||||||
|
],
|
||||||
|
"cc": [
|
||||||
|
"https://example.com/users/rat/followers"
|
||||||
|
],
|
||||||
|
"content": "wants to read",
|
||||||
|
"replies": {
|
||||||
|
"id": "https://example.com/users/rat/generatednote/2567/replies",
|
||||||
|
"type": "OrderedCollection",
|
||||||
|
"totalItems": 0,
|
||||||
|
"first": "https://example.com/users/rat/generatednote/2567/replies?page=true",
|
||||||
|
"last": "https://example.com/users/rat/generatednote/2567/replies?page=true",
|
||||||
|
"name": "",
|
||||||
|
"@context": "https://www.w3.org/ns/activitystreams"
|
||||||
|
},
|
||||||
|
"tag": [
|
||||||
|
{
|
||||||
|
"href": "https://bookwyrm.social/book/37292",
|
||||||
|
"name": "Female Husbands",
|
||||||
|
"type": "Book"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"attachment": [],
|
||||||
|
"sensitive": false,
|
||||||
|
"@context": "https://www.w3.org/ns/activitystreams"
|
||||||
|
},
|
||||||
|
"to": [
|
||||||
|
"https://example.com/users/rat/followers"
|
||||||
|
],
|
||||||
|
"cc": [
|
||||||
|
"https://www.w3.org/ns/activitystreams#Public"
|
||||||
|
],
|
||||||
|
"signature": {
|
||||||
|
"creator": "https://example.com/users/rat#main-key",
|
||||||
|
"created": "2020-12-16T01:45:19.662734+00:00",
|
||||||
|
"signatureValue": "R+W8nN1CQAlREjSUeaQwJXZrXTOOLvpHQi9n/3vd8QKq+l6HJEpu7eAht9fjpk8YOKEgV3OUQ7w3E42wM4t+sFiaPoQjY6Xy9IOvx/2LcOZjSOtTkiZ1XnnVb3DSbl8BOBH02+cPvoR6k4LIPHm2IHYZ1UL02WdDWaicHEwl7bw=",
|
||||||
|
"type": "RsaSignature2017"
|
||||||
|
},
|
||||||
|
"@context": "https://www.w3.org/ns/activitystreams"
|
||||||
|
}
|
|
@ -1 +0,0 @@
|
||||||
from . import *
|
|
|
@ -1,80 +0,0 @@
|
||||||
from unittest.mock import patch
|
|
||||||
from django.test import TestCase
|
|
||||||
|
|
||||||
from bookwyrm import models, outgoing
|
|
||||||
from bookwyrm.settings import DOMAIN
|
|
||||||
|
|
||||||
|
|
||||||
class Following(TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
with patch('bookwyrm.models.user.set_remote_server'):
|
|
||||||
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',
|
|
||||||
)
|
|
||||||
self.local_user = models.User.objects.create_user(
|
|
||||||
'mouse', 'mouse@mouse.com', 'mouseword',
|
|
||||||
local=True,
|
|
||||||
remote_id='http://local.com/users/mouse',
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_follow(self):
|
|
||||||
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
|
||||||
outgoing.handle_follow(self.local_user, self.remote_user)
|
|
||||||
|
|
||||||
rel = models.UserFollowRequest.objects.get()
|
|
||||||
|
|
||||||
self.assertEqual(rel.user_subject, self.local_user)
|
|
||||||
self.assertEqual(rel.user_object, self.remote_user)
|
|
||||||
self.assertEqual(rel.status, 'follow_request')
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_unfollow(self):
|
|
||||||
self.remote_user.followers.add(self.local_user)
|
|
||||||
self.assertEqual(self.remote_user.followers.count(), 1)
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
|
||||||
outgoing.handle_unfollow(self.local_user, self.remote_user)
|
|
||||||
|
|
||||||
self.assertEqual(self.remote_user.followers.count(), 0)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_accept(self):
|
|
||||||
rel = models.UserFollowRequest.objects.create(
|
|
||||||
user_subject=self.local_user,
|
|
||||||
user_object=self.remote_user
|
|
||||||
)
|
|
||||||
rel_id = rel.id
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
|
||||||
outgoing.handle_accept(rel)
|
|
||||||
# request should be deleted
|
|
||||||
self.assertEqual(
|
|
||||||
models.UserFollowRequest.objects.filter(id=rel_id).count(), 0
|
|
||||||
)
|
|
||||||
# follow relationship should exist
|
|
||||||
self.assertEqual(self.remote_user.followers.first(), self.local_user)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_reject(self):
|
|
||||||
rel = models.UserFollowRequest.objects.create(
|
|
||||||
user_subject=self.local_user,
|
|
||||||
user_object=self.remote_user
|
|
||||||
)
|
|
||||||
rel_id = rel.id
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
|
||||||
outgoing.handle_reject(rel)
|
|
||||||
# request should be deleted
|
|
||||||
self.assertEqual(
|
|
||||||
models.UserFollowRequest.objects.filter(id=rel_id).count(), 0
|
|
||||||
)
|
|
||||||
# follow relationship should not exist
|
|
||||||
self.assertEqual(
|
|
||||||
models.UserFollows.objects.filter(id=rel_id).count(), 0
|
|
||||||
)
|
|
|
@ -1,61 +0,0 @@
|
||||||
''' testing user lookup '''
|
|
||||||
import json
|
|
||||||
import pathlib
|
|
||||||
from unittest.mock import patch
|
|
||||||
|
|
||||||
from django.test import TestCase
|
|
||||||
import responses
|
|
||||||
|
|
||||||
from bookwyrm import models, outgoing
|
|
||||||
from bookwyrm.settings import DOMAIN
|
|
||||||
|
|
||||||
class TestOutgoingRemoteWebfinger(TestCase):
|
|
||||||
''' overwrites standard model feilds to work with activitypub '''
|
|
||||||
def setUp(self):
|
|
||||||
''' get user data ready '''
|
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath(
|
|
||||||
'../data/ap_user.json'
|
|
||||||
)
|
|
||||||
self.userdata = json.loads(datafile.read_bytes())
|
|
||||||
del self.userdata['icon']
|
|
||||||
|
|
||||||
def test_existing_user(self):
|
|
||||||
''' simple database lookup by username '''
|
|
||||||
user = models.User.objects.create_user(
|
|
||||||
'mouse', 'mouse@mouse.mouse', 'mouseword', local=True)
|
|
||||||
|
|
||||||
result = outgoing.handle_remote_webfinger('@mouse@%s' % DOMAIN)
|
|
||||||
self.assertEqual(result, user)
|
|
||||||
|
|
||||||
result = outgoing.handle_remote_webfinger('mouse@%s' % DOMAIN)
|
|
||||||
self.assertEqual(result, user)
|
|
||||||
|
|
||||||
|
|
||||||
@responses.activate
|
|
||||||
def test_load_user(self):
|
|
||||||
username = 'mouse@example.com'
|
|
||||||
wellknown = {
|
|
||||||
"subject": "acct:mouse@example.com",
|
|
||||||
"links": [
|
|
||||||
{
|
|
||||||
"rel": "self",
|
|
||||||
"type": "application/activity+json",
|
|
||||||
"href": "https://example.com/user/mouse"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
responses.add(
|
|
||||||
responses.GET,
|
|
||||||
'https://example.com/.well-known/webfinger?resource=acct:%s' \
|
|
||||||
% username,
|
|
||||||
json=wellknown,
|
|
||||||
status=200)
|
|
||||||
responses.add(
|
|
||||||
responses.GET,
|
|
||||||
'https://example.com/user/mouse',
|
|
||||||
json=self.userdata,
|
|
||||||
status=200)
|
|
||||||
with patch('bookwyrm.models.user.set_remote_server.delay'):
|
|
||||||
result = outgoing.handle_remote_webfinger('@mouse@example.com')
|
|
||||||
self.assertIsInstance(result, models.User)
|
|
||||||
self.assertEqual(result.username, 'mouse@example.com')
|
|
|
@ -1,69 +0,0 @@
|
||||||
from unittest.mock import patch
|
|
||||||
from django.test import TestCase
|
|
||||||
|
|
||||||
from bookwyrm import models, outgoing
|
|
||||||
|
|
||||||
|
|
||||||
class Shelving(TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
self.user = models.User.objects.create_user(
|
|
||||||
'mouse', 'mouse@mouse.com', 'mouseword',
|
|
||||||
local=True,
|
|
||||||
remote_id='http://local.com/users/mouse',
|
|
||||||
)
|
|
||||||
work = models.Work.objects.create(
|
|
||||||
title='Example work',
|
|
||||||
)
|
|
||||||
self.book = models.Edition.objects.create(
|
|
||||||
title='Example Edition',
|
|
||||||
remote_id='https://example.com/book/1',
|
|
||||||
parent_work=work,
|
|
||||||
)
|
|
||||||
self.shelf = models.Shelf.objects.create(
|
|
||||||
name='Test Shelf',
|
|
||||||
identifier='test-shelf',
|
|
||||||
user=self.user
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_shelve(self):
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay') as _:
|
|
||||||
outgoing.handle_shelve(self.user, self.book, self.shelf)
|
|
||||||
# make sure the book is on the shelf
|
|
||||||
self.assertEqual(self.shelf.books.get(), self.book)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_shelve_to_read(self):
|
|
||||||
shelf = models.Shelf.objects.get(identifier='to-read')
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay') as _:
|
|
||||||
outgoing.handle_shelve(self.user, self.book, shelf)
|
|
||||||
# make sure the book is on the shelf
|
|
||||||
self.assertEqual(shelf.books.get(), self.book)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_shelve_reading(self):
|
|
||||||
shelf = models.Shelf.objects.get(identifier='reading')
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay') as _:
|
|
||||||
outgoing.handle_shelve(self.user, self.book, shelf)
|
|
||||||
# make sure the book is on the shelf
|
|
||||||
self.assertEqual(shelf.books.get(), self.book)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_shelve_read(self):
|
|
||||||
shelf = models.Shelf.objects.get(identifier='read')
|
|
||||||
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay') as _:
|
|
||||||
outgoing.handle_shelve(self.user, self.book, shelf)
|
|
||||||
# make sure the book is on the shelf
|
|
||||||
self.assertEqual(shelf.books.get(), self.book)
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_unshelve(self):
|
|
||||||
self.shelf.books.add(self.book)
|
|
||||||
self.shelf.save()
|
|
||||||
self.assertEqual(self.shelf.books.count(), 1)
|
|
||||||
with patch('bookwyrm.broadcast.broadcast_task.delay') as _:
|
|
||||||
outgoing.handle_unshelve(self.user, self.book, self.shelf)
|
|
||||||
self.assertEqual(self.shelf.books.count(), 0)
|
|
|
@ -8,6 +8,7 @@ from django.http import HttpResponseBadRequest, HttpResponseNotAllowed, \
|
||||||
HttpResponseNotFound
|
HttpResponseNotFound
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
from django.test.client import RequestFactory
|
from django.test.client import RequestFactory
|
||||||
|
import responses
|
||||||
|
|
||||||
from bookwyrm import models, incoming
|
from bookwyrm import models, incoming
|
||||||
|
|
||||||
|
@ -421,6 +422,25 @@ class Incoming(TestCase):
|
||||||
self.assertEqual(notification.related_status, self.status)
|
self.assertEqual(notification.related_status, self.status)
|
||||||
|
|
||||||
|
|
||||||
|
@responses.activate
|
||||||
|
def test_handle_discarded_boost(self):
|
||||||
|
''' test a boost of a mastodon status that will be discarded '''
|
||||||
|
activity = {
|
||||||
|
'type': 'Announce',
|
||||||
|
'id': 'http://www.faraway.com/boost/12',
|
||||||
|
'actor': self.remote_user.remote_id,
|
||||||
|
'object': self.status.to_activity(),
|
||||||
|
}
|
||||||
|
responses.add(
|
||||||
|
responses.GET,
|
||||||
|
'http://www.faraway.com/boost/12',
|
||||||
|
json={'id': 'http://www.faraway.com/boost/12'},
|
||||||
|
status=200)
|
||||||
|
incoming.handle_boost(activity)
|
||||||
|
self.assertEqual(models.Boost.objects.count(), 0)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def test_handle_unboost(self):
|
def test_handle_unboost(self):
|
||||||
''' undo a boost '''
|
''' undo a boost '''
|
||||||
activity = {
|
activity = {
|
||||||
|
|
193
bookwyrm/tests/test_outgoing.py
Normal file
193
bookwyrm/tests/test_outgoing.py
Normal file
|
@ -0,0 +1,193 @@
|
||||||
|
''' sending out activities '''
|
||||||
|
import json
|
||||||
|
import pathlib
|
||||||
|
from unittest.mock import patch
|
||||||
|
|
||||||
|
from django.test import TestCase
|
||||||
|
import responses
|
||||||
|
|
||||||
|
from bookwyrm import models, outgoing
|
||||||
|
from bookwyrm.settings import DOMAIN
|
||||||
|
|
||||||
|
|
||||||
|
class Outgoing(TestCase):
|
||||||
|
''' sends out activities '''
|
||||||
|
def setUp(self):
|
||||||
|
''' we'll need some data '''
|
||||||
|
with patch('bookwyrm.models.user.set_remote_server'):
|
||||||
|
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',
|
||||||
|
)
|
||||||
|
self.local_user = models.User.objects.create_user(
|
||||||
|
'mouse', 'mouse@mouse.com', 'mouseword', local=True,
|
||||||
|
remote_id='https://example.com/users/mouse',
|
||||||
|
)
|
||||||
|
|
||||||
|
datafile = pathlib.Path(__file__).parent.joinpath(
|
||||||
|
'data/ap_user.json'
|
||||||
|
)
|
||||||
|
self.userdata = json.loads(datafile.read_bytes())
|
||||||
|
del self.userdata['icon']
|
||||||
|
|
||||||
|
work = models.Work.objects.create(title='Test Work')
|
||||||
|
self.book = models.Edition.objects.create(
|
||||||
|
title='Example Edition',
|
||||||
|
remote_id='https://example.com/book/1',
|
||||||
|
parent_work=work
|
||||||
|
)
|
||||||
|
self.shelf = models.Shelf.objects.create(
|
||||||
|
name='Test Shelf',
|
||||||
|
identifier='test-shelf',
|
||||||
|
user=self.local_user
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_follow(self):
|
||||||
|
''' send a follow request '''
|
||||||
|
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_follow(self.local_user, self.remote_user)
|
||||||
|
|
||||||
|
rel = models.UserFollowRequest.objects.get()
|
||||||
|
|
||||||
|
self.assertEqual(rel.user_subject, self.local_user)
|
||||||
|
self.assertEqual(rel.user_object, self.remote_user)
|
||||||
|
self.assertEqual(rel.status, 'follow_request')
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_unfollow(self):
|
||||||
|
''' send an unfollow '''
|
||||||
|
self.remote_user.followers.add(self.local_user)
|
||||||
|
self.assertEqual(self.remote_user.followers.count(), 1)
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_unfollow(self.local_user, self.remote_user)
|
||||||
|
|
||||||
|
self.assertEqual(self.remote_user.followers.count(), 0)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_accept(self):
|
||||||
|
''' accept a follow request '''
|
||||||
|
rel = models.UserFollowRequest.objects.create(
|
||||||
|
user_subject=self.local_user,
|
||||||
|
user_object=self.remote_user
|
||||||
|
)
|
||||||
|
rel_id = rel.id
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_accept(rel)
|
||||||
|
# request should be deleted
|
||||||
|
self.assertEqual(
|
||||||
|
models.UserFollowRequest.objects.filter(id=rel_id).count(), 0
|
||||||
|
)
|
||||||
|
# follow relationship should exist
|
||||||
|
self.assertEqual(self.remote_user.followers.first(), self.local_user)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_reject(self):
|
||||||
|
''' reject a follow request '''
|
||||||
|
rel = models.UserFollowRequest.objects.create(
|
||||||
|
user_subject=self.local_user,
|
||||||
|
user_object=self.remote_user
|
||||||
|
)
|
||||||
|
rel_id = rel.id
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_reject(rel)
|
||||||
|
# request should be deleted
|
||||||
|
self.assertEqual(
|
||||||
|
models.UserFollowRequest.objects.filter(id=rel_id).count(), 0
|
||||||
|
)
|
||||||
|
# follow relationship should not exist
|
||||||
|
self.assertEqual(
|
||||||
|
models.UserFollows.objects.filter(id=rel_id).count(), 0
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_existing_user(self):
|
||||||
|
''' simple database lookup by username '''
|
||||||
|
result = outgoing.handle_remote_webfinger('@mouse@%s' % DOMAIN)
|
||||||
|
self.assertEqual(result, self.local_user)
|
||||||
|
|
||||||
|
result = outgoing.handle_remote_webfinger('mouse@%s' % DOMAIN)
|
||||||
|
self.assertEqual(result, self.local_user)
|
||||||
|
|
||||||
|
|
||||||
|
@responses.activate
|
||||||
|
def test_load_user(self):
|
||||||
|
''' find a remote user using webfinger '''
|
||||||
|
username = 'mouse@example.com'
|
||||||
|
wellknown = {
|
||||||
|
"subject": "acct:mouse@example.com",
|
||||||
|
"links": [{
|
||||||
|
"rel": "self",
|
||||||
|
"type": "application/activity+json",
|
||||||
|
"href": "https://example.com/user/mouse"
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
responses.add(
|
||||||
|
responses.GET,
|
||||||
|
'https://example.com/.well-known/webfinger?resource=acct:%s' \
|
||||||
|
% username,
|
||||||
|
json=wellknown,
|
||||||
|
status=200)
|
||||||
|
responses.add(
|
||||||
|
responses.GET,
|
||||||
|
'https://example.com/user/mouse',
|
||||||
|
json=self.userdata,
|
||||||
|
status=200)
|
||||||
|
with patch('bookwyrm.models.user.set_remote_server.delay'):
|
||||||
|
result = outgoing.handle_remote_webfinger('@mouse@example.com')
|
||||||
|
self.assertIsInstance(result, models.User)
|
||||||
|
self.assertEqual(result.username, 'mouse@example.com')
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_shelve(self):
|
||||||
|
''' shelve a book '''
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_shelve(self.local_user, self.book, self.shelf)
|
||||||
|
# make sure the book is on the shelf
|
||||||
|
self.assertEqual(self.shelf.books.get(), self.book)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_shelve_to_read(self):
|
||||||
|
''' special behavior for the to-read shelf '''
|
||||||
|
shelf = models.Shelf.objects.get(identifier='to-read')
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_shelve(self.local_user, self.book, shelf)
|
||||||
|
# make sure the book is on the shelf
|
||||||
|
self.assertEqual(shelf.books.get(), self.book)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_shelve_reading(self):
|
||||||
|
''' special behavior for the reading shelf '''
|
||||||
|
shelf = models.Shelf.objects.get(identifier='reading')
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_shelve(self.local_user, self.book, shelf)
|
||||||
|
# make sure the book is on the shelf
|
||||||
|
self.assertEqual(shelf.books.get(), self.book)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_shelve_read(self):
|
||||||
|
''' special behavior for the read shelf '''
|
||||||
|
shelf = models.Shelf.objects.get(identifier='read')
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_shelve(self.local_user, self.book, shelf)
|
||||||
|
# make sure the book is on the shelf
|
||||||
|
self.assertEqual(shelf.books.get(), self.book)
|
||||||
|
|
||||||
|
|
||||||
|
def test_handle_unshelve(self):
|
||||||
|
''' remove a book from a shelf '''
|
||||||
|
self.shelf.books.add(self.book)
|
||||||
|
self.shelf.save()
|
||||||
|
self.assertEqual(self.shelf.books.count(), 1)
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
outgoing.handle_unshelve(self.local_user, self.book, self.shelf)
|
||||||
|
self.assertEqual(self.shelf.books.count(), 0)
|
|
@ -6,7 +6,7 @@ from django.http.response import Http404
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
from django.test.client import RequestFactory
|
from django.test.client import RequestFactory
|
||||||
|
|
||||||
from bookwyrm import view_actions as actions, models
|
from bookwyrm import forms, models, view_actions as actions
|
||||||
from bookwyrm.settings import DOMAIN
|
from bookwyrm.settings import DOMAIN
|
||||||
|
|
||||||
|
|
||||||
|
@ -238,6 +238,43 @@ class ViewActions(TestCase):
|
||||||
self.assertEqual(resp.template_name, 'password_reset.html')
|
self.assertEqual(resp.template_name, 'password_reset.html')
|
||||||
self.assertTrue(models.PasswordReset.objects.exists())
|
self.assertTrue(models.PasswordReset.objects.exists())
|
||||||
|
|
||||||
|
|
||||||
|
def test_password_change(self):
|
||||||
|
''' change password '''
|
||||||
|
password_hash = self.local_user.password
|
||||||
|
request = self.factory.post('', {
|
||||||
|
'password': 'hi',
|
||||||
|
'confirm-password': 'hi'
|
||||||
|
})
|
||||||
|
request.user = self.local_user
|
||||||
|
with patch('bookwyrm.view_actions.login'):
|
||||||
|
resp = actions.password_change(request)
|
||||||
|
self.assertNotEqual(self.local_user.password, password_hash)
|
||||||
|
|
||||||
|
def test_password_change_mismatch(self):
|
||||||
|
''' change password '''
|
||||||
|
password_hash = self.local_user.password
|
||||||
|
request = self.factory.post('', {
|
||||||
|
'password': 'hi',
|
||||||
|
'confirm-password': 'hihi'
|
||||||
|
})
|
||||||
|
request.user = self.local_user
|
||||||
|
resp = actions.password_change(request)
|
||||||
|
self.assertEqual(self.local_user.password, password_hash)
|
||||||
|
|
||||||
|
|
||||||
|
def test_edit_user(self):
|
||||||
|
''' use a form to update a user '''
|
||||||
|
form = forms.EditUserForm(instance=self.local_user)
|
||||||
|
form.data['name'] = 'New Name'
|
||||||
|
request = self.factory.post('', form.data)
|
||||||
|
request.user = self.local_user
|
||||||
|
|
||||||
|
with patch('bookwyrm.broadcast.broadcast_task.delay'):
|
||||||
|
actions.edit_profile(request)
|
||||||
|
self.assertEqual(self.local_user.name, 'New Name')
|
||||||
|
|
||||||
|
|
||||||
def test_switch_edition(self):
|
def test_switch_edition(self):
|
||||||
''' updates user's relationships to a book '''
|
''' updates user's relationships to a book '''
|
||||||
work = models.Work.objects.create(title='test work')
|
work = models.Work.objects.create(title='test work')
|
||||||
|
|
|
@ -159,23 +159,21 @@ def password_change(request):
|
||||||
request.user.set_password(new_password)
|
request.user.set_password(new_password)
|
||||||
request.user.save()
|
request.user.save()
|
||||||
login(request, request.user)
|
login(request, request.user)
|
||||||
return redirect('/user-edit')
|
return redirect('/user/%s' % request.user.localname)
|
||||||
|
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
@require_POST
|
@require_POST
|
||||||
def edit_profile(request):
|
def edit_profile(request):
|
||||||
''' les get fancy with images '''
|
''' les get fancy with images '''
|
||||||
form = forms.EditUserForm(request.POST, request.FILES)
|
form = forms.EditUserForm(
|
||||||
|
request.POST, request.FILES, instance=request.user)
|
||||||
if not form.is_valid():
|
if not form.is_valid():
|
||||||
data = {
|
data = {'form': form, 'user': request.user}
|
||||||
'form': form,
|
|
||||||
'user': request.user,
|
|
||||||
}
|
|
||||||
return TemplateResponse(request, 'edit_user.html', data)
|
return TemplateResponse(request, 'edit_user.html', data)
|
||||||
|
|
||||||
request.user.name = form.data['name']
|
user = form.save(commit=False)
|
||||||
request.user.email = form.data['email']
|
|
||||||
if 'avatar' in form.files:
|
if 'avatar' in form.files:
|
||||||
# crop and resize avatar upload
|
# crop and resize avatar upload
|
||||||
image = Image.open(form.files['avatar'])
|
image = Image.open(form.files['avatar'])
|
||||||
|
@ -201,17 +199,10 @@ def edit_profile(request):
|
||||||
# set the name to a hash
|
# set the name to a hash
|
||||||
extension = form.files['avatar'].name.split('.')[-1]
|
extension = form.files['avatar'].name.split('.')[-1]
|
||||||
filename = '%s.%s' % (uuid4(), extension)
|
filename = '%s.%s' % (uuid4(), extension)
|
||||||
request.user.avatar.save(
|
user.avatar.save(filename, ContentFile(output.getvalue()))
|
||||||
filename,
|
user.save()
|
||||||
ContentFile(output.getvalue())
|
|
||||||
)
|
|
||||||
|
|
||||||
request.user.summary = form.data['summary']
|
outgoing.handle_update_user(user)
|
||||||
request.user.manually_approves_followers = \
|
|
||||||
form.cleaned_data['manually_approves_followers']
|
|
||||||
request.user.save()
|
|
||||||
|
|
||||||
outgoing.handle_update_user(request.user)
|
|
||||||
return redirect('/user/%s' % request.user.localname)
|
return redirect('/user/%s' % request.user.localname)
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue