bookwyrm-mastodon/bookwyrm/tests/views/books/test_book.py

457 lines
16 KiB
Python
Raw Normal View History

2021-03-08 11:49:10 -05:00
""" test for app action functionality """
from io import BytesIO
import pathlib
from unittest.mock import patch
from tidylib import tidy_document
from PIL import Image
import responses
from django.contrib.auth.models import Group, Permission
from django.contrib.contenttypes.models import ContentType
from django.core.files.uploadedfile import SimpleUploadedFile
2021-09-27 15:26:56 -04:00
from django.http import Http404
from django.template.response import TemplateResponse
from django.test import TestCase
from django.test.client import RequestFactory
2021-08-06 13:19:38 -04:00
from django.utils import timezone
from bookwyrm import forms, models, views
from bookwyrm.activitypub import ActivitypubResponse
2021-01-13 13:04:44 -05:00
class BookViews(TestCase):
2021-04-26 12:15:42 -04:00
"""books books books"""
2021-03-08 11:49:10 -05:00
def setUp(self):
2021-04-26 12:15:42 -04:00
"""we need basic test data and mocks"""
self.factory = RequestFactory()
with patch("bookwyrm.suggested_users.rerank_suggestions_task.delay"), patch(
"bookwyrm.activitystreams.populate_stream_task.delay"
):
2021-08-03 13:25:53 -04:00
self.local_user = models.User.objects.create_user(
"mouse@local.com",
"mouse@mouse.com",
"mouseword",
local=True,
localname="mouse",
remote_id="https://example.com/users/mouse",
)
2021-03-08 11:49:10 -05:00
self.group = Group.objects.create(name="editor")
self.group.permissions.add(
Permission.objects.create(
2021-03-08 11:49:10 -05:00
name="edit_book",
codename="edit_book",
content_type=ContentType.objects.get_for_model(models.User),
).id
)
2021-08-02 19:05:40 -04:00
self.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=self.work,
)
models.SiteSettings.objects.create()
def test_book_page(self):
2021-04-26 12:15:42 -04:00
"""there are so many views, this just makes sure it LOADS"""
view = views.Book.as_view()
2021-08-06 13:19:38 -04:00
models.ReadThrough.objects.create(
user=self.local_user,
book=self.book,
start_date=timezone.now(),
)
2021-03-08 11:49:10 -05:00
request = self.factory.get("")
request.user = self.local_user
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.book.id)
self.assertIsInstance(result, TemplateResponse)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
# idk how else to filter out these unescape amp errs
2021-09-30 10:47:19 -04:00
errors = "\n".join(
e
for e in errors.split("\n")
if "&book" not in e and "id and name attribute" not in e
)
if errors:
raise Exception(errors)
self.assertEqual(result.status_code, 200)
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
is_api.return_value = True
result = view(request, self.book.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
2021-08-06 13:19:38 -04:00
@patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay")
2021-09-06 16:53:49 -04:00
@patch("bookwyrm.activitystreams.add_status_task.delay")
2021-08-06 20:43:50 -04:00
def test_book_page_statuses(self, *_):
2021-08-06 13:19:38 -04:00
"""there are so many views, this just makes sure it LOADS"""
view = views.Book.as_view()
review = models.Review.objects.create(
user=self.local_user,
book=self.book,
content="hi",
)
comment = models.Comment.objects.create(
user=self.local_user,
book=self.book,
content="hi",
)
quote = models.Quotation.objects.create(
user=self.local_user,
book=self.book,
content="hi",
quote="wow",
)
request = self.factory.get("")
request.user = self.local_user
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
2021-08-06 13:19:38 -04:00
is_api.return_value = False
result = view(request, self.book.id, user_statuses="review")
self.assertIsInstance(result, TemplateResponse)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
2021-09-30 10:47:19 -04:00
errors = "\n".join(
e
for e in errors.split("\n")
if "&book" not in e and "id and name attribute" not in e
)
if errors:
raise Exception(errors)
2021-08-06 13:19:38 -04:00
self.assertEqual(result.status_code, 200)
self.assertEqual(result.context_data["statuses"].object_list[0], review)
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
2021-08-06 13:19:38 -04:00
is_api.return_value = False
result = view(request, self.book.id, user_statuses="comment")
self.assertIsInstance(result, TemplateResponse)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
if errors:
raise Exception(errors)
2021-08-06 13:19:38 -04:00
self.assertEqual(result.status_code, 200)
self.assertEqual(result.context_data["statuses"].object_list[0], comment)
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
2021-08-06 13:19:38 -04:00
is_api.return_value = False
result = view(request, self.book.id, user_statuses="quotation")
self.assertIsInstance(result, TemplateResponse)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
if errors:
raise Exception(errors)
2021-08-06 13:19:38 -04:00
self.assertEqual(result.status_code, 200)
self.assertEqual(result.context_data["statuses"].object_list[0], quote)
def test_book_page_invalid_id(self):
"""there are so many views, this just makes sure it LOADS"""
view = views.Book.as_view()
request = self.factory.get("")
request.user = self.local_user
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
2021-08-06 13:19:38 -04:00
is_api.return_value = False
2021-09-27 15:26:56 -04:00
with self.assertRaises(Http404):
view(request, 0)
2021-08-06 13:19:38 -04:00
def test_book_page_work_id(self):
"""there are so many views, this just makes sure it LOADS"""
view = views.Book.as_view()
request = self.factory.get("")
request.user = self.local_user
with patch("bookwyrm.views.books.books.is_api_request") as is_api:
2021-08-06 13:19:38 -04:00
is_api.return_value = False
result = view(request, self.work.id)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
if errors:
raise Exception(errors)
2021-08-06 13:19:38 -04:00
self.assertEqual(result.status_code, 200)
self.assertEqual(result.context_data["book"], self.book)
def test_edit_book_page(self):
2021-04-26 12:15:42 -04:00
"""there are so many views, this just makes sure it LOADS"""
view = views.EditBook.as_view()
2021-03-08 11:49:10 -05:00
request = self.factory.get("")
request.user = self.local_user
request.user.is_superuser = True
2021-08-02 19:05:40 -04:00
result = view(request, self.book.id)
self.assertIsInstance(result, TemplateResponse)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
if errors:
raise Exception(errors)
self.assertEqual(result.status_code, 200)
def test_edit_book(self):
2021-04-26 12:15:42 -04:00
"""lets a user edit a book"""
view = views.EditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm(instance=self.book)
2021-03-08 11:49:10 -05:00
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
request = self.factory.post("", form.data)
request.user = self.local_user
2021-05-26 17:57:29 -04:00
2021-08-02 19:05:40 -04:00
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.book.id)
2021-05-26 17:57:29 -04:00
self.book.refresh_from_db()
2021-03-08 11:49:10 -05:00
self.assertEqual(self.book.title, "New Title")
2021-03-04 16:48:50 -05:00
def test_edit_book_add_author(self):
2021-04-26 12:15:42 -04:00
"""lets a user edit a book with new authors"""
2021-03-07 16:59:27 -05:00
view = views.EditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm(instance=self.book)
2021-03-08 13:10:30 -05:00
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
form.data["add_author"] = "Sappho"
request = self.factory.post("", form.data)
2021-03-07 16:59:27 -05:00
request.user = self.local_user
2021-08-02 19:05:40 -04:00
result = view(request, self.book.id)
html = result.render()
_, errors = tidy_document(
html.content,
options={
"drop-empty-elements": False,
"warn-proprietary-attributes": False,
},
)
if errors:
raise Exception(errors)
2021-03-07 16:59:27 -05:00
# the changes haven't been saved yet
self.book.refresh_from_db()
2021-03-08 13:10:30 -05:00
self.assertEqual(self.book.title, "Example Edition")
2021-03-07 16:59:27 -05:00
def test_edit_book_add_new_author_confirm(self):
2021-04-26 12:15:42 -04:00
"""lets a user edit a book confirmed with new authors"""
2021-03-07 16:59:27 -05:00
view = views.ConfirmEditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm(instance=self.book)
2021-03-08 13:10:30 -05:00
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
2021-03-13 21:09:09 -05:00
form.data["author-match-count"] = 1
form.data["author_match-0"] = "Sappho"
2021-03-08 13:10:30 -05:00
request = self.factory.post("", form.data)
2021-03-07 16:59:27 -05:00
request.user = self.local_user
2021-08-02 19:05:40 -04:00
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.book.id)
2021-03-07 16:59:27 -05:00
self.book.refresh_from_db()
2021-03-08 13:10:30 -05:00
self.assertEqual(self.book.title, "New Title")
self.assertEqual(self.book.authors.first().name, "Sappho")
2021-03-04 16:48:50 -05:00
2021-03-07 18:14:57 -05:00
def test_edit_book_remove_author(self):
2021-04-26 12:15:42 -04:00
"""remove an author from a book"""
2021-03-08 13:10:30 -05:00
author = models.Author.objects.create(name="Sappho")
2021-03-07 18:14:57 -05:00
self.book.authors.add(author)
form = forms.EditionForm(instance=self.book)
view = views.EditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm(instance=self.book)
2021-03-08 13:10:30 -05:00
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
form.data["remove_authors"] = [author.id]
request = self.factory.post("", form.data)
2021-03-07 18:14:57 -05:00
request.user = self.local_user
2021-08-02 19:05:40 -04:00
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.book.id)
2021-03-07 18:14:57 -05:00
self.book.refresh_from_db()
2021-03-08 13:10:30 -05:00
self.assertEqual(self.book.title, "New Title")
2021-03-07 18:14:57 -05:00
self.assertFalse(self.book.authors.exists())
2021-03-08 13:48:45 -05:00
def test_create_book(self):
2021-04-26 12:15:42 -04:00
"""create an entirely new book and work"""
2021-03-08 13:48:45 -05:00
view = views.ConfirmEditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm()
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
request = self.factory.post("", form.data)
request.user = self.local_user
2021-08-02 19:05:40 -04:00
view(request)
2021-03-08 13:48:45 -05:00
book = models.Edition.objects.get(title="New Title")
self.assertEqual(book.parent_work.title, "New Title")
def test_create_book_existing_work(self):
2021-04-26 12:15:42 -04:00
"""create an entirely new book and work"""
2021-03-08 13:48:45 -05:00
view = views.ConfirmEditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm()
form.data["title"] = "New Title"
form.data["parent_work"] = self.work.id
form.data["last_edited_by"] = self.local_user.id
request = self.factory.post("", form.data)
request.user = self.local_user
2021-08-02 19:05:40 -04:00
view(request)
2021-03-08 13:48:45 -05:00
book = models.Edition.objects.get(title="New Title")
self.assertEqual(book.parent_work, self.work)
def test_create_book_with_author(self):
2021-04-26 12:15:42 -04:00
"""create an entirely new book and work"""
view = views.ConfirmEditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm()
form.data["title"] = "New Title"
2021-03-13 21:09:09 -05:00
form.data["author-match-count"] = "1"
form.data["author_match-0"] = "Sappho"
form.data["last_edited_by"] = self.local_user.id
request = self.factory.post("", form.data)
request.user = self.local_user
2021-08-02 19:05:40 -04:00
view(request)
book = models.Edition.objects.get(title="New Title")
self.assertEqual(book.parent_work.title, "New Title")
self.assertEqual(book.authors.first().name, "Sappho")
self.assertEqual(book.authors.first(), book.parent_work.authors.first())
def _setup_cover_url(self): # pylint: disable=no-self-use
"""creates cover url mock"""
cover_url = "http://example.com"
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)
responses.add(
responses.GET,
cover_url,
body=output.getvalue(),
status=200,
)
return cover_url
@responses.activate
def test_create_book_upload_cover_url(self):
"""create an entirely new book and work with cover url"""
self.assertFalse(self.book.cover)
self.local_user.groups.add(self.group)
cover_url = self._setup_cover_url()
form = forms.EditionForm()
form.data["title"] = "New Title"
form.data["last_edited_by"] = self.local_user.id
form.data["cover-url"] = cover_url
request = self.factory.post("", form.data)
request.user = self.local_user
with patch(
"bookwyrm.models.activitypub_mixin.broadcast_task.delay"
) as delay_mock:
views.upload_cover(request, self.book.id)
self.assertEqual(delay_mock.call_count, 1)
self.book.refresh_from_db()
self.assertTrue(self.book.cover)
def test_upload_cover_file(self):
2021-04-26 12:15:42 -04:00
"""add a cover via file upload"""
self.assertFalse(self.book.cover)
image_file = pathlib.Path(__file__).parent.joinpath(
"../../static/images/default_avi.jpg"
)
form = forms.CoverForm(instance=self.book)
# pylint: disable=consider-using-with
form.data["cover"] = SimpleUploadedFile(
image_file, open(image_file, "rb").read(), content_type="image/jpeg"
)
request = self.factory.post("", form.data)
request.user = self.local_user
2021-08-02 19:05:40 -04:00
with patch(
"bookwyrm.models.activitypub_mixin.broadcast_task.delay"
) as delay_mock:
views.upload_cover(request, self.book.id)
self.assertEqual(delay_mock.call_count, 1)
self.book.refresh_from_db()
self.assertTrue(self.book.cover)
@responses.activate
def test_upload_cover_url(self):
2021-04-26 12:15:42 -04:00
"""add a cover via url"""
self.assertFalse(self.book.cover)
form = forms.CoverForm(instance=self.book)
form.data["cover-url"] = self._setup_cover_url()
request = self.factory.post("", form.data)
request.user = self.local_user
2021-08-02 19:05:40 -04:00
with patch(
"bookwyrm.models.activitypub_mixin.broadcast_task.delay"
) as delay_mock:
views.upload_cover(request, self.book.id)
self.assertEqual(delay_mock.call_count, 1)
self.book.refresh_from_db()
self.assertTrue(self.book.cover)
def test_add_description(self):
"""add a book description"""
self.local_user.groups.add(self.group)
request = self.factory.post("", {"description": "new description hi"})
request.user = self.local_user
2021-08-05 21:58:32 -04:00
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.add_description(request, self.book.id)
self.book.refresh_from_db()
self.assertEqual(self.book.description, "new description hi")
self.assertEqual(self.book.last_edited_by, self.local_user)