- {% if request.user.is_authenticated %}
+ {% if request.user.is_authenticated %}
-
+
{{ user.display_name }}
+
@@ -141,12 +149,12 @@
{% trans "Username:" %}
-
+
{% trans "Log in" %}
@@ -157,7 +165,7 @@
{% if site.allow_registration and request.path != '' and request.path != '/' %}
{% endif %}
@@ -191,11 +199,11 @@
{% if site.support_link %}
- Support {{ site.name }} on
{{ site.support_title }}
+ {% blocktrans with site_name=site.name support_link=site.support_link support_title=site.support_title %}Support {{ site_name }} on
{{ support_title }} {% endblocktrans %}
{% endif %}
- BookWyrm is open source software. You can contribute or report issues on
GitHub .
+ {% trans 'BookWyrm is open source software. You can contribute or report issues on
GitHub .' %}
diff --git a/bookwyrm/templates/lists/created_text.html b/bookwyrm/templates/lists/created_text.html
new file mode 100644
index 00000000..eee5a75f
--- /dev/null
+++ b/bookwyrm/templates/lists/created_text.html
@@ -0,0 +1,10 @@
+{% load i18n %}
+{% spaceless %}
+
+{% if list.curation != 'open' %}
+{% blocktrans with username=list.user.display_name path=list.user.local_path %}Created and curated by
{{ username }} {% endblocktrans %}
+{% else %}
+{% blocktrans with username=list.user.display_name path=list.user.local_path %}Created by
{{ username }} {% endblocktrans %}
+{% endif %}
+
+{% endspaceless %}
diff --git a/bookwyrm/templates/lists/curate.html b/bookwyrm/templates/lists/curate.html
index a040dd5f..f4ee147a 100644
--- a/bookwyrm/templates/lists/curate.html
+++ b/bookwyrm/templates/lists/curate.html
@@ -24,7 +24,7 @@
{% include 'snippets/book_titleby.html' with book=item.book %}
- {% include 'snippets/username.html' with user=item.user %}
+ {{ item.user.display_name }}
diff --git a/bookwyrm/templates/lists/list.html b/bookwyrm/templates/lists/list.html
index 42d99d8a..ddac04f3 100644
--- a/bookwyrm/templates/lists/list.html
+++ b/bookwyrm/templates/lists/list.html
@@ -1,8 +1,8 @@
{% extends 'lists/list_layout.html' %}
{% load i18n %}
{% load bookwyrm_tags %}
-{% block panel %}
+{% block panel %}
{% if request.user == list.user and pending_count %}
diff --git a/bookwyrm/templates/lists/list_layout.html b/bookwyrm/templates/lists/list_layout.html
index 8e2e36e6..c0899c84 100644
--- a/bookwyrm/templates/lists/list_layout.html
+++ b/bookwyrm/templates/lists/list_layout.html
@@ -1,18 +1,22 @@
{% extends 'layout.html' %}
{% load i18n %}
{% load bookwyrm_tags %}
-{% block content %}
+{% block title %}{{ list.name }}{% endblock %}
+
+{% block content %}
{{ list.name }} {% include 'snippets/privacy-icons.html' with item=list %}
-
{% if list.curation != 'open' %}{% trans "Created and curated by" %}{% else %}{% trans "Created by" %} {% include 'snippets/username.html' with user=list.user %}
- {% endif %}
+
+ {% include 'lists/created_text.html' with list=list %}
+
{% include 'snippets/trimmed_text.html' with full=list.description %}
{% if request.user == list.user %}
- {% include 'snippets/toggle/open_button.html' with text="Edit list" icon="pencil" controls_text="edit-list" focus="edit-list-header" %}
+ {% trans "Edit List" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text icon="pencil" controls_text="edit-list" focus="edit-list-header" %}
{% endif %}
diff --git a/bookwyrm/templates/lists/lists.html b/bookwyrm/templates/lists/lists.html
index 07032676..f7ab020a 100644
--- a/bookwyrm/templates/lists/lists.html
+++ b/bookwyrm/templates/lists/lists.html
@@ -1,5 +1,8 @@
{% extends 'layout.html' %}
{% load i18n %}
+
+{% block title %}{% trans "Lists" %}{% endblock %}
+
{% block content %}
@@ -11,7 +14,8 @@
{% trans "Your lists" %}
- {% include 'snippets/toggle/open_button.html' with controls_text="create-list" icon="plus" text="Create new list" focus="create-list-header" %}
+ {% trans "Create List" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with controls_text="create-list" icon="plus" text=button_text focus="create-list-header" %}
@@ -25,7 +29,7 @@
{% endif %}
{% if request.user.list_set.count > 4 %}
- See all {{ request.user.list_set.count}} lists
+ {% blocktrans with size=request.user.list_set.count %}See all {{ size }} lists{% endblocktrans %}
{% endif %}
{% endif %}
diff --git a/bookwyrm/templates/login.html b/bookwyrm/templates/login.html
index 1429f412..063b4b6d 100644
--- a/bookwyrm/templates/login.html
+++ b/bookwyrm/templates/login.html
@@ -1,7 +1,9 @@
{% extends 'layout.html' %}
{% load i18n %}
-{% block content %}
+{% block title %}{% trans "Login" %}{% endblock %}
+
+{% block content %}
diff --git a/bookwyrm/templates/notfound.html b/bookwyrm/templates/notfound.html
index bb419cad..210e1a13 100644
--- a/bookwyrm/templates/notfound.html
+++ b/bookwyrm/templates/notfound.html
@@ -1,10 +1,11 @@
{% extends 'layout.html' %}
{% load i18n %}
-{% block content %}
+{% block title %}{% trans "Not Found" %}{% endblock %}
+
+{% block content %}
{% trans "Not Found" %}
-
{% trans "The page your requested doesn't seem to exist!" %}
+
{% trans "The page you requested doesn't seem to exist!" %}
-
{% endblock %}
diff --git a/bookwyrm/templates/notifications.html b/bookwyrm/templates/notifications.html
index ea9f24b8..80ee2250 100644
--- a/bookwyrm/templates/notifications.html
+++ b/bookwyrm/templates/notifications.html
@@ -2,6 +2,9 @@
{% load i18n %}
{% load humanize %}
{% load bookwyrm_tags %}
+
+{% block title %}{% trans "Notifications" %}{% endblock %}
+
{% block content %}
{% trans "Notifications" %}
@@ -39,16 +42,42 @@
{# DESCRIPTION #}
{% if notification.related_user %}
- {% include 'snippets/avatar.html' with user=notification.related_user %}
- {% include 'snippets/username.html' with user=notification.related_user %}
+
+ {% include 'snippets/avatar.html' with user=notification.related_user %}
+ {{ notification.related_user.display_name }}
+
{% if notification.notification_type == 'FAVORITE' %}
- {% blocktrans with preview_name=related_status|status_preview_name|safe related_path=related_status.local_path %}favorited your {{ preview_name }} {% endblocktrans %}
+ {% if related_status.status_type == 'Review' %}
+ {% blocktrans with book_title=related_status.book.title related_path=related_status.local_path %}favorited your review of {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Comment' %}
+ {% blocktrans with book_title=related_status.book.title related_path=related_status.local_path %}favorited your comment on {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Quotation' %}
+ {% blocktrans with book_title=related_status.book.title related_path=related_status.local_path %}favorited your quote from {{ book_title }} {% endblocktrans %}
+ {% else %}
+ {% blocktrans with related_path=related_status.local_path %}favorited your status {% endblocktrans %}
+ {% endif %}
{% elif notification.notification_type == 'MENTION' %}
- {% blocktrans with preview_name=related_status|status_preview_name|safe related_path=related_status.local_path %}mentioned you in a {{ preview_name }} {% endblocktrans %}
+ {% if related_status.status_type == 'Review' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}mentioned you in a review of {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Comment' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}mentioned you in a comment on {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Quotation' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}mentioned you in a quote from {{ book_title }} {% endblocktrans %}
+ {% else %}
+ {% blocktrans with related_path=related_status.local_path %}mentioned you in a status {% endblocktrans %}
+ {% endif %}
{% elif notification.notification_type == 'REPLY' %}
- {% blocktrans with preview_name=related_status|status_preview_name|safe related_path=related_status.local_path parent_path=related_status.reply_parent.local_path %}replied to your {{ preview_name }} {% endblocktrans %}
+ {% if related_status.status_type == 'Review' %}
+ {% blocktrans with related_path=related_status.local_path parent_path=related_status.reply_parent.local_path book_title=related_status.reply_parent.book.title %}replied to your review of {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Comment' %}
+ {% blocktrans with related_path=related_status.local_path parent_path=related_status.reply_parent.local_path book_title=related_status.reply_parent.book.title %}replied to your comment on {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Quotation' %}
+ {% blocktrans with related_path=related_status.local_path parent_path=related_status.reply_parent.local_path book_title=related_status.reply_parent.book.title %}replied to your quote from {{ book_title }} {% endblocktrans %}
+ {% else %}
+ {% blocktrans with related_path=related_status.local_path parent_path=related_status.reply_parent.local_path %}replied to your status {% endblocktrans %}
+ {% endif %}
{% elif notification.notification_type == 'FOLLOW' %}
{% trans "followed you" %}
@@ -59,9 +88,21 @@
{% include 'snippets/follow_request_buttons.html' with user=notification.related_user %}
{% elif notification.notification_type == 'BOOST' %}
- {% blocktrans with preview_name=related_status|status_preview_name|safe related_path=related_status.local_path %}boosted your
{{ preview_name }} {% endblocktrans %}
+ {% if related_status.status_type == 'Review' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}boosted your
review of {{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Comment' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}boosted your
comment on{{ book_title }} {% endblocktrans %}
+ {% elif related_status.status_type == 'Quotation' %}
+ {% blocktrans with related_path=related_status.local_path book_title=related_status.book.title %}boosted your
quote from {{ book_title }} {% endblocktrans %}
+ {% else %}
+ {% blocktrans with related_path=related_status.local_path %}boosted your
status {% endblocktrans %}
+ {% endif %}
{% elif notification.notification_type == 'ADD' %}
- {% if notification.related_list_item.approved %}{% trans "added" %}{% else %}{% trans "suggested adding" %}{% endif %} {% include 'snippets/book_titleby.html' with book=notification.related_list_item.book %} to your list "
{{ notification.related_list_item.book_list.name }} "
+ {% if notification.related_list_item.approved %}
+ {% blocktrans with book_path=notification.related_list_item.book.local_path book_title=notification.related_list_item.book.title list_path=notification.related_list_item.book_list.local_path list_name=notification.related_list_item.book_list.name %} added
{{ book_title }} to your list "
{{ list_name }} "{% endblocktrans %}
+ {% else %}
+ {% blocktrans with book_path=notification.related_list_item.book.local_path book_title=notification.related_list_item.book.title list_path=notification.related_list_item.book_list.local_path list_name=notification.related_list_item.book_list.name %} suggested adding
{{ book_title }} to your list "
{{ list_name }} "{% endblocktrans %}
+ {% endif %}
{% endif %}
{% elif notification.related_import %}
{% blocktrans with related_id=notification.related_import.id %} your
import completed.{% endblocktrans %}
@@ -75,7 +116,9 @@
{% if related_status.content %}
-
{{ related_status.content | safe | truncatewords_html:10 }}
+
+ {{ related_status.content | safe | truncatewords_html:10 }}{% if related_status.mention_books %} {{ related_status.mention_books.first.title }} {% endif %}
+
{% elif related_status.quote %}
{{ related_status.quote | safe | truncatewords_html:10 }}
{% elif related_status.rating %}
diff --git a/bookwyrm/templates/password_reset.html b/bookwyrm/templates/password_reset.html
index 656e2dff..57add722 100644
--- a/bookwyrm/templates/password_reset.html
+++ b/bookwyrm/templates/password_reset.html
@@ -1,7 +1,9 @@
{% extends 'layout.html' %}
{% load i18n %}
-{% block content %}
+{% block title %}{% trans "Reset Password" %}{% endblock %}
+
+{% block content %}
diff --git a/bookwyrm/templates/password_reset_request.html b/bookwyrm/templates/password_reset_request.html
index 87dc8e65..97191afa 100644
--- a/bookwyrm/templates/password_reset_request.html
+++ b/bookwyrm/templates/password_reset_request.html
@@ -1,7 +1,9 @@
{% extends 'layout.html' %}
{% load i18n %}
-{% block content %}
+{% block title %}{% trans "Reset Password" %}{% endblock %}
+
+{% block content %}
diff --git a/bookwyrm/templates/preferences/blocks.html b/bookwyrm/templates/preferences/blocks.html
index e26a5229..72d4abd4 100644
--- a/bookwyrm/templates/preferences/blocks.html
+++ b/bookwyrm/templates/preferences/blocks.html
@@ -1,6 +1,8 @@
{% extends 'preferences/preferences_layout.html' %}
{% load i18n %}
+{% block title %}{% trans "Blocked Users" %}{{ author.name }}{% endblock %}
+
{% block header %}
{% trans "Blocked Users" %}
{% endblock %}
@@ -13,7 +15,7 @@
{% for user in request.user.blocks.all %}
- {% include 'snippets/avatar.html' with user=user %} {% include 'snippets/username.html' with user=user %}
+ {% include 'snippets/avatar.html' with user=user %} {{ user.display_name }}
{% include 'snippets/block_button.html' with user=user %}
diff --git a/bookwyrm/templates/preferences/change_password.html b/bookwyrm/templates/preferences/change_password.html
index 46707946..ab8be717 100644
--- a/bookwyrm/templates/preferences/change_password.html
+++ b/bookwyrm/templates/preferences/change_password.html
@@ -1,5 +1,8 @@
{% extends 'preferences/preferences_layout.html' %}
{% load i18n %}
+
+{% block title %}{% trans "Change Password" %}{% endblock %}
+
{% block header %}
{% trans "Change Password" %}
{% endblock %}
@@ -15,6 +18,6 @@
{% trans "Confirm password:" %}
-
{% trans "Change password" %}
+
{% trans "Change Password" %}
{% endblock %}
diff --git a/bookwyrm/templates/preferences/edit_user.html b/bookwyrm/templates/preferences/edit_user.html
index d5b76c07..b643f4ce 100644
--- a/bookwyrm/templates/preferences/edit_user.html
+++ b/bookwyrm/templates/preferences/edit_user.html
@@ -1,5 +1,8 @@
{% extends 'preferences/preferences_layout.html' %}
{% load i18n %}
+
+{% block title %}{% trans "Edit Profile" %}{% endblock %}
+
{% block header %}
{% trans "Edit Profile" %}
{% endblock %}
diff --git a/bookwyrm/templates/preferences/preferences_layout.html b/bookwyrm/templates/preferences/preferences_layout.html
index c0f78f95..d463d9cb 100644
--- a/bookwyrm/templates/preferences/preferences_layout.html
+++ b/bookwyrm/templates/preferences/preferences_layout.html
@@ -14,13 +14,13 @@
{% trans "Profile" %}
- {% trans "Change password" %}
+ {% trans "Change Password" %}
diff --git a/bookwyrm/templates/search_results.html b/bookwyrm/templates/search_results.html
index 30c88f3d..4e8481f0 100644
--- a/bookwyrm/templates/search_results.html
+++ b/bookwyrm/templates/search_results.html
@@ -1,5 +1,8 @@
{% extends 'layout.html' %}
{% load i18n %}
+
+{% block title %}{% trans "Search Results" %}{% endblock %}
+
{% block content %}
{% with book_results|first as local_results %}
@@ -29,7 +32,8 @@
{% trans "Didn't find what you were looking for?" %}
- {% include 'snippets/toggle/open_button.html' with text="Show results from other catalogues" small=True controls_text="more-results" %}
+ {% trans "Show results from other catalogues" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text small=True controls_text="more-results" %}
{% endif %}
@@ -60,7 +64,8 @@
{% endfor %}
{% if local_results.results %}
- {% include 'snippets/toggle/close_button.html' with text="Hide results from other catalogues" small=True controls_text="more-results" %}
+ {% trans "Hide results from other catalogues" as button_text %}
+ {% include 'snippets/toggle/close_button.html' with text=button_text small=True controls_text="more-results" %}
{% endif %}
{% endif %}
@@ -74,8 +79,10 @@
{% for result in user_results %}
- {% include 'snippets/avatar.html' with user=result %}
- {% include 'snippets/username.html' with user=result show_full=True %}
+
+ {% include 'snippets/avatar.html' with user=result %}
+ {{ result.display_name }}
+ ({{ result.username }})
{% include 'snippets/follow_button.html' with user=result %}
{% endfor %}
diff --git a/bookwyrm/templates/settings/admin_layout.html b/bookwyrm/templates/settings/admin_layout.html
index 73178ac8..16741436 100644
--- a/bookwyrm/templates/settings/admin_layout.html
+++ b/bookwyrm/templates/settings/admin_layout.html
@@ -1,5 +1,8 @@
{% extends 'layout.html' %}
{% load i18n %}
+
+{% block title %}{% trans "Administration" %}{% endblock %}
+
{% block content %}
{% endblock %}
diff --git a/bookwyrm/templates/snippets/status/status.html b/bookwyrm/templates/snippets/status/status.html
index b8925f18..10492c48 100644
--- a/bookwyrm/templates/snippets/status/status.html
+++ b/bookwyrm/templates/snippets/status/status.html
@@ -1,9 +1,11 @@
{% load bookwyrm_tags %}
{% load i18n %}
{% if not status.deleted %}
- {% if status.status_type == 'Boost' %}
- {% include 'snippets/avatar.html' with user=status.user %}
- {% include 'snippets/username.html' with user=status.user %}
+ {% if status.status_type == 'Announce' %}
+
+ {% include 'snippets/avatar.html' with user=status.user %}
+ {{ status.user.display_name }}
+
{% trans "boosted" %}
{% include 'snippets/status/status_body.html' with status=status|boosted_status %}
{% else %}
diff --git a/bookwyrm/templates/snippets/status/status_body.html b/bookwyrm/templates/snippets/status/status_body.html
index 1bf0d3c7..8d6c21ed 100644
--- a/bookwyrm/templates/snippets/status/status_body.html
+++ b/bookwyrm/templates/snippets/status/status_body.html
@@ -21,7 +21,8 @@
{% if request.user.is_authenticated %}
- {% include 'snippets/toggle/toggle_button.html' with controls_text="show-comment" controls_uid=status.id text="Reply" icon="comment" class="is-small toggle-button" focus="id_content_reply" %}
+ {% trans "Reply" as button_text %}
+ {% include 'snippets/toggle/toggle_button.html' with controls_text="show-comment" controls_uid=status.id text=button_text icon="comment" class="is-small toggle-button" focus="id_content_reply" %}
{% include 'snippets/boost_button.html' with status=status %}
diff --git a/bookwyrm/templates/snippets/status/status_content.html b/bookwyrm/templates/snippets/status/status_content.html
index 9c79f3a5..616e1e63 100644
--- a/bookwyrm/templates/snippets/status/status_content.html
+++ b/bookwyrm/templates/snippets/status/status_content.html
@@ -3,23 +3,27 @@
{% if status.status_type == 'Review' or status.status_type == 'Rating' %}
-
- {% if status.name %}{{ status.name }} {% endif %}
+ {% if status.name %}
+
+ {{ status.name|escape }}
- {% include 'snippets/stars.html' with rating=status.rating %}
+ {% endif %}
+ {% include 'snippets/stars.html' with rating=status.rating %}
{% endif %}
{% if status.content_warning %}
{{ status.content_warning }}
- {% include 'snippets/toggle/open_button.html' with text="show more" class="is-small" controls_text="show-status-cw" controls_uid=status.id %}
+ {% trans "Show more" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text class="is-small" controls_text="show-status-cw" controls_uid=status.id %}
{% endif %}
{% if status.content_warning %}
- {% include 'snippets/toggle/close_button.html' with text="show less" class="is-small" controls_text="show-status-cw" controls_uid=status.id %}
+ {% trans "Show less" as button_text %}
+ {% include 'snippets/toggle/close_button.html' with text=button_text class="is-small" controls_text="show-status-cw" controls_uid=status.id %}
{% endif %}
{% if status.quote %}
@@ -30,10 +34,10 @@
{% endif %}
- {% if status.content and status.status_type != 'GeneratedNote' and status.status_type != 'Boost' %}
+ {% if status.content and status.status_type != 'GeneratedNote' and status.status_type != 'Announce' %}
{% include 'snippets/trimmed_text.html' with full=status.content|safe %}
{% endif %}
- {% if status.attachments %}
+ {% if status.attachments.exists %}
{% for attachment in status.attachments.all %}
diff --git a/bookwyrm/templates/snippets/status/status_header.html b/bookwyrm/templates/snippets/status/status_header.html
index a07c0f35..a04177d9 100644
--- a/bookwyrm/templates/snippets/status/status_header.html
+++ b/bookwyrm/templates/snippets/status/status_header.html
@@ -1,7 +1,9 @@
{% load bookwyrm_tags %}
{% load i18n %}
-{% include 'snippets/avatar.html' with user=status.user %}
-{% include 'snippets/username.html' with user=status.user %}
+
+{% include 'snippets/avatar.html' with user=status.user ariaHide="true" %}
+{{ status.user.display_name }}
+
{% if status.status_type == 'GeneratedNote' %}
{{ status.content | safe }}
@@ -15,7 +17,17 @@
{% trans "quoted" %}
{% elif status.reply_parent %}
{% with parent_status=status|parent %}
- replied to {% include 'snippets/username.html' with user=parent_status.user possessive=True %}
{% if parent_status.status_type == 'GeneratedNote' %}update{% else %}{{ parent_status.status_type | lower }}{% endif %}
+
+ {% if parent_status.status_type == 'Review' %}
+ {% blocktrans with username=parent_status.user.display_name user_path=parent_status.user.local_path status_path=parent_status.local_path %}replied to
{{ username}}'s review {% endblocktrans %}
+ {% elif parent_status.status_type == 'Comment' %}
+ {% blocktrans with username=parent_status.user.display_name user_path=parent_status.user.local_path status_path=parent_status.local_path %}replied to
{{ username}}'s comment {% endblocktrans %}
+ {% elif parent_status.status_type == 'Quotation' %}
+ {% blocktrans with username=parent_status.user.display_name user_path=parent_status.user.local_path status_path=parent_status.local_path %}replied to
{{ username}}'s quote {% endblocktrans %}
+ {% else %}
+ {% blocktrans with username=parent_status.user.display_name user_path=parent_status.user.local_path status_path=parent_status.local_path %}replied to
{{ username}}'s status {% endblocktrans %}
+ {% endif %}
+
{% endwith %}
{% endif %}
{% if status.book %}
diff --git a/bookwyrm/templates/snippets/status/status_options.html b/bookwyrm/templates/snippets/status/status_options.html
index 735ff51c..babd8296 100644
--- a/bookwyrm/templates/snippets/status/status_options.html
+++ b/bookwyrm/templates/snippets/status/status_options.html
@@ -14,7 +14,7 @@
diff --git a/bookwyrm/templates/snippets/trimmed_text.html b/bookwyrm/templates/snippets/trimmed_text.html
index b349a4fc..4ec92bd6 100644
--- a/bookwyrm/templates/snippets/trimmed_text.html
+++ b/bookwyrm/templates/snippets/trimmed_text.html
@@ -1,4 +1,6 @@
{% load bookwyrm_tags %}
+{% load i18n %}
+
{% with 0|uuid as uuid %}
{% if full %}
{% with full|to_markdown|safe as full %}
@@ -6,17 +8,29 @@
{% with full|to_markdown|safe|truncatewords_html:60 as trimmed %}
{% if trimmed != full %}
-
{{ trimmed }}
- {% include 'snippets/toggle/open_button.html' with text="show more" controls_text="full" controls_uid=uuid class="is-small" %}
+
+
{{ trimmed }}
+
+
+ {% trans "Show more" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text controls_text="full" controls_uid=uuid class="is-small" %}
+
-
{{ full }}
- {% include 'snippets/toggle/close_button.html' with text="show less" controls_text="full" controls_uid=uuid class="is-small" %}
+
+
{{ full }}
+
+
+ {% trans "Show less" as button_text %}
+ {% include 'snippets/toggle/close_button.html' with text=button_text controls_text="full" controls_uid=uuid class="is-small" %}
+
{% else %}
-
{{ full }}
+
{% endif %}
{% endwith %}
diff --git a/bookwyrm/templates/snippets/username.html b/bookwyrm/templates/snippets/username.html
deleted file mode 100644
index a6f4b2b0..00000000
--- a/bookwyrm/templates/snippets/username.html
+++ /dev/null
@@ -1,2 +0,0 @@
-{% load bookwyrm_tags %}
-
{% if user.name %}{{ user.name }}{% else %}{{ user | username }}{% endif %} {% if possessive %}'s{% endif %}{% if show_full and user.name or show_full and user.localname %} ({{ user.username }}){% endif %}
diff --git a/bookwyrm/templates/tag.html b/bookwyrm/templates/tag.html
index f30b7395..b6fa6778 100644
--- a/bookwyrm/templates/tag.html
+++ b/bookwyrm/templates/tag.html
@@ -1,13 +1,14 @@
{% extends 'layout.html' %}
{% load i18n %}
{% load bookwyrm_tags %}
-{% block content %}
+{% block title %}{{ tag.name }}{% endblock %}
+
+{% block content %}
{% blocktrans %}Books tagged "{{ tag.name }}"{% endblocktrans %}
{% include 'snippets/book_tiles.html' with books=books.all %}
-
{% endblock %}
diff --git a/bookwyrm/templates/user/create_shelf_form.html b/bookwyrm/templates/user/create_shelf_form.html
index 785c8d06..b7ea27de 100644
--- a/bookwyrm/templates/user/create_shelf_form.html
+++ b/bookwyrm/templates/user/create_shelf_form.html
@@ -2,7 +2,7 @@
{% load i18n %}
{% block header %}
-{% trans "Create New Shelf" %}
+{% trans "Create Shelf" %}
{% endblock %}
{% block form %}
@@ -19,7 +19,7 @@
{% include 'snippets/privacy_select.html' %}
- {% trans "Create shelf" %}
+ {% trans "Create Shelf" %}
diff --git a/bookwyrm/templates/user/edit_shelf_form.html b/bookwyrm/templates/user/edit_shelf_form.html
index a9f86da4..753d0681 100644
--- a/bookwyrm/templates/user/edit_shelf_form.html
+++ b/bookwyrm/templates/user/edit_shelf_form.html
@@ -29,4 +29,3 @@
{% endblock %}
-
diff --git a/bookwyrm/templates/user/followers.html b/bookwyrm/templates/user/followers.html
index 16ec7d4f..45d87a3d 100644
--- a/bookwyrm/templates/user/followers.html
+++ b/bookwyrm/templates/user/followers.html
@@ -4,30 +4,29 @@
{% block header %}
- {% if is_self %}Your
- {% else %}
- {% include 'snippets/username.html' with user=user possessive=True %}
- {% endif %}
- followers
+ {% trans "User Profile" %}
{% endblock %}
{% block panel %}
{% trans "Followers" %}
- {% for followers in followers %}
+ {% for follower in followers %}
- {% include 'snippets/follow_button.html' with user=followers %}
+ {% include 'snippets/follow_button.html' with user=follower %}
{% endfor %}
{% if not followers.count %}
-
{% blocktrans with username=user|username %}{{ username }} has no followers{% endblocktrans %}
+
{% blocktrans with username=user.display_name %}{{ username }} has no followers{% endblocktrans %}
{% endif %}
{% endblock %}
diff --git a/bookwyrm/templates/user/following.html b/bookwyrm/templates/user/following.html
index 13af4417..5904c1bb 100644
--- a/bookwyrm/templates/user/following.html
+++ b/bookwyrm/templates/user/following.html
@@ -4,11 +4,7 @@
{% block header %}
- Users following
- {% if is_self %}you
- {% else %}
- {% include 'snippets/username.html' with user=user %}
- {% endif %}
+ {% trans "User Profile" %}
{% endblock %}
@@ -18,8 +14,11 @@
{% for follower in user.following.all %}
{% include 'snippets/follow_button.html' with user=follower %}
diff --git a/bookwyrm/templates/user/lists.html b/bookwyrm/templates/user/lists.html
index 6edf4430..85c7cc8c 100644
--- a/bookwyrm/templates/user/lists.html
+++ b/bookwyrm/templates/user/lists.html
@@ -5,16 +5,17 @@
- {% if is_self %}Your
+ {% if is_self %}
+ {% trans "Your Lists" %}
{% else %}
- {% include 'snippets/username.html' with user=user %}'s
+ {% blocktrans with username=user.display_name %}Lists: {{ username }}{% endblocktrans %}
{% endif %}
- Lists
{% if is_self %}
- {% include 'snippets/toggle/open_button.html' with controls_text="create-list" icon="plus" text="Create new list" %}
+ {% trans "Create list" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with controls_text="create-list" icon="plus" text=button_text %}
{% endif %}
diff --git a/bookwyrm/templates/user/shelf.html b/bookwyrm/templates/user/shelf.html
index e62e2218..189d2856 100644
--- a/bookwyrm/templates/user/shelf.html
+++ b/bookwyrm/templates/user/shelf.html
@@ -1,14 +1,15 @@
{% extends 'user/user_layout.html' %}
{% load bookwyrm_tags %}
+{% load i18n %}
{% block header %}
- {% if is_self %}Your
+ {% if is_self %}
+ {% trans "Your Shelves" %}
{% else %}
- {% include 'snippets/username.html' with user=user possessive=True %}
+ {% blocktrans with username=user.display_name %}{{ username }}: Shelves{% endblocktrans %}
{% endif %}
- shelves
{% endblock %}
@@ -16,20 +17,21 @@
{% block panel %}
-
{% if is_self %}
- {% include 'snippets/toggle/open_button.html' with text="Create shelf" icon="plus" controls_text="create-shelf-form" focus="create-shelf-form-header" %}
+ {% trans "Create shelf" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text icon="plus" controls_text="create-shelf-form" focus="create-shelf-form-header" %}
{% endif %}
@@ -49,7 +51,8 @@
{% if is_self %}
- {% include 'snippets/toggle/open_button.html' with text="Edit shelf" icon="pencil" controls_text="edit-shelf-form" focus="edit-shelf-form-header" %}
+ {% trans "Edit shelf" as button_text %}
+ {% include 'snippets/toggle/open_button.html' with text=button_text icon="pencil" controls_text="edit-shelf-form" focus="edit-shelf-form-header" %}
{% endif %}
diff --git a/bookwyrm/templates/user/user.html b/bookwyrm/templates/user/user.html
index bcab8d84..52a91561 100644
--- a/bookwyrm/templates/user/user.html
+++ b/bookwyrm/templates/user/user.html
@@ -1,10 +1,12 @@
{% extends 'user/user_layout.html' %}
{% load i18n %}
+{% block title %}{{ user.display_name }}{% endblock %}
+
{% block header %}
-
{% trans "User profile" %}
+ {% trans "User Profile" %}
{% if is_self %}
@@ -65,7 +67,7 @@
{% for activity in activities %}
-
+
{% include 'snippets/status/status.html' with status=activity %}
{% endfor %}
diff --git a/bookwyrm/templates/user/user_layout.html b/bookwyrm/templates/user/user_layout.html
index c8eea412..d60db2a0 100644
--- a/bookwyrm/templates/user/user_layout.html
+++ b/bookwyrm/templates/user/user_layout.html
@@ -3,6 +3,8 @@
{% load humanize %}
{% load bookwyrm_tags %}
+{% block title %}{{ user.display_name }}{% endblock %}
+
{% block content %}
{% block header %}{% endblock %}
@@ -15,11 +17,11 @@
{% include 'user/user_preview.html' with user=user %}
+ {% if user.summary %}
- {% if user.summary %}
{{ user.summary | to_markdown | safe }}
- {% endif %}
+ {% endif %}
{% if not is_self and request.user.is_authenticated %}
{% include 'snippets/follow_button.html' with user=user %}
@@ -31,7 +33,7 @@
{% for requester in user.follower_requests.all %}
- {% include 'snippets/username.html' with user=requester show_full=True %}
+ {{ requester.display_name }} ({{ requester.username }})
{% include 'snippets/follow_request_buttons.html' with user=requester %}
@@ -54,7 +56,7 @@
{% trans "Reading Goal" %}
{% endif %}
- {% if is_self or user.lists.exists %}
+ {% if is_self or user.list_set.exists %}
{% url 'user-lists' user|username as url %}
{% trans "Lists" %}
diff --git a/bookwyrm/templates/user/user_preview.html b/bookwyrm/templates/user/user_preview.html
index 3158ff54..c641c58f 100644
--- a/bookwyrm/templates/user/user_preview.html
+++ b/bookwyrm/templates/user/user_preview.html
@@ -12,8 +12,8 @@
{{ user.username }}
{% blocktrans with date=user.created_date|naturaltime %}Joined {{ date }}{% endblocktrans %}
- {{ user.followers.count }} follower{{ user.followers.count | pluralize }} ,
- {{ user.following.count }} following
+ {% blocktrans count counter=user.followers.count %}{{ counter }} follower{% plural %}{{ counter }} followers{% endblocktrans %} ,
+ {% blocktrans with counter=user.following.count %}{{ counter }} following{% endblocktrans %}
diff --git a/bookwyrm/templatetags/bookwyrm_tags.py b/bookwyrm/templatetags/bookwyrm_tags.py
index 67354ac6..77be0ca3 100644
--- a/bookwyrm/templatetags/bookwyrm_tags.py
+++ b/bookwyrm/templatetags/bookwyrm_tags.py
@@ -1,4 +1,4 @@
-''' template filters '''
+""" template filters """
from uuid import uuid4
from datetime import datetime
@@ -13,66 +13,78 @@ from bookwyrm.views.status import to_markdown
register = template.Library()
-@register.filter(name='dict_key')
+
+@register.filter(name="dict_key")
def dict_key(d, k):
- ''' Returns the given key from a dictionary. '''
+ """ Returns the given key from a dictionary. """
return d.get(k) or 0
-@register.filter(name='rating')
+@register.filter(name="rating")
def get_rating(book, user):
- ''' get the overall rating of a book '''
+ """ get the overall rating of a book """
queryset = views.helpers.privacy_filter(
- user, models.Review.objects.filter(book=book))
- return queryset.aggregate(Avg('rating'))['rating__avg']
+ user, models.Review.objects.filter(book=book)
+ )
+ return queryset.aggregate(Avg("rating"))["rating__avg"]
-@register.filter(name='user_rating')
+@register.filter(name="user_rating")
def get_user_rating(book, user):
- ''' get a user's rating of a book '''
- rating = models.Review.objects.filter(
- user=user,
- book=book,
- rating__isnull=False,
- ).order_by('-published_date').first()
+ """ get a user's rating of a book """
+ rating = (
+ models.Review.objects.filter(
+ user=user,
+ book=book,
+ rating__isnull=False,
+ )
+ .order_by("-published_date")
+ .first()
+ )
if rating:
return rating.rating
return 0
-@register.filter(name='username')
+@register.filter(name="username")
def get_user_identifier(user):
- ''' use localname for local users, username for remote '''
+ """ use localname for local users, username for remote """
return user.localname if user.localname else user.username
-@register.filter(name='notification_count')
+@register.filter(name="notification_count")
def get_notification_count(user):
- ''' how many UNREAD notifications are there '''
+ """ how many UNREAD notifications are there """
return user.notification_set.filter(read=False).count()
-@register.filter(name='replies')
+@register.filter(name="replies")
def get_replies(status):
- ''' get all direct replies to a status '''
- #TODO: this limit could cause problems
- return models.Status.objects.filter(
- reply_parent=status,
- deleted=False,
- ).select_subclasses().all()[:10]
+ """ get all direct replies to a status """
+ # TODO: this limit could cause problems
+ return (
+ models.Status.objects.filter(
+ reply_parent=status,
+ deleted=False,
+ )
+ .select_subclasses()
+ .all()[:10]
+ )
-@register.filter(name='parent')
+@register.filter(name="parent")
def get_parent(status):
- ''' get the reply parent for a status '''
- return models.Status.objects.filter(
- id=status.reply_parent_id
- ).select_subclasses().get()
+ """ get the reply parent for a status """
+ return (
+ models.Status.objects.filter(id=status.reply_parent_id)
+ .select_subclasses()
+ .get()
+ )
-@register.filter(name='liked')
+@register.filter(name="liked")
def get_user_liked(user, status):
- ''' did the given user fav a status? '''
+ """ did the given user fav a status? """
try:
models.Favorite.objects.get(user=user, status=status)
return True
@@ -80,15 +92,15 @@ def get_user_liked(user, status):
return False
-@register.filter(name='boosted')
+@register.filter(name="boosted")
def get_user_boosted(user, status):
- ''' did the given user fav a status? '''
- return user.id in status.boosters.all().values_list('user', flat=True)
+ """ did the given user fav a status? """
+ return user.id in status.boosters.all().values_list("user", flat=True)
-@register.filter(name='follow_request_exists')
+@register.filter(name="follow_request_exists")
def follow_request_exists(user, requester):
- ''' see if there is a pending follow request for a user '''
+ """ see if there is a pending follow request for a user """
try:
models.UserFollowRequest.objects.filter(
user_subject=requester,
@@ -99,129 +111,139 @@ def follow_request_exists(user, requester):
return False
-@register.filter(name='boosted_status')
+@register.filter(name="boosted_status")
def get_boosted(boost):
- ''' load a boosted status. have to do this or it wont get foregin keys '''
- return models.Status.objects.select_subclasses().filter(
- id=boost.boosted_status.id
- ).get()
+ """ load a boosted status. have to do this or it wont get foregin keys """
+ return (
+ models.Status.objects.select_subclasses()
+ .filter(id=boost.boosted_status.id)
+ .get()
+ )
-@register.filter(name='book_description')
+@register.filter(name="book_description")
def get_book_description(book):
- ''' use the work's text if the book doesn't have it '''
+ """ use the work's text if the book doesn't have it """
return book.description or book.parent_work.description
-@register.filter(name='uuid')
+@register.filter(name="uuid")
def get_uuid(identifier):
- ''' for avoiding clashing ids when there are many forms '''
- return '%s%s' % (identifier, uuid4())
+ """ for avoiding clashing ids when there are many forms """
+ return "%s%s" % (identifier, uuid4())
-@register.filter(name='post_date')
+@register.filter(name="post_date")
def time_since(date):
- ''' concise time ago function '''
+ """ concise time ago function """
if not isinstance(date, datetime):
- return ''
+ return ""
now = timezone.now()
delta = now - date
if date < (now - relativedelta(weeks=1)):
- formatter = '%b %-d'
+ formatter = "%b %-d"
if date.year != now.year:
- formatter += ' %Y'
+ formatter += " %Y"
return date.strftime(formatter)
delta = relativedelta(now, date)
if delta.days:
- return '%dd' % delta.days
+ return "%dd" % delta.days
if delta.hours:
- return '%dh' % delta.hours
+ return "%dh" % delta.hours
if delta.minutes:
- return '%dm' % delta.minutes
- return '%ds' % delta.seconds
+ return "%dm" % delta.minutes
+ return "%ds" % delta.seconds
-@register.filter(name='to_markdown')
+@register.filter(name="to_markdown")
def get_markdown(content):
- ''' convert markdown to html '''
+ """ convert markdown to html """
if content:
return to_markdown(content)
return None
-@register.filter(name='mentions')
-def get_mentions(status, user):
- ''' people to @ in a reply: the parent and all mentions '''
- mentions = set([status.user] + list(status.mention_users.all()))
- return ' '.join(
- '@' + get_user_identifier(m) for m in mentions if not m == user) + ' '
-@register.filter(name='status_preview_name')
+@register.filter(name="mentions")
+def get_mentions(status, user):
+ """ people to @ in a reply: the parent and all mentions """
+ mentions = set([status.user] + list(status.mention_users.all()))
+ return (
+ " ".join("@" + get_user_identifier(m) for m in mentions if not m == user) + " "
+ )
+
+
+@register.filter(name="status_preview_name")
def get_status_preview_name(obj):
- ''' text snippet with book context for a status '''
+ """ text snippet with book context for a status """
name = obj.__class__.__name__.lower()
- if name == 'review':
- return '%s of
%s ' % (name, obj.book.title)
- if name == 'comment':
- return '%s on
%s ' % (name, obj.book.title)
- if name == 'quotation':
- return '%s from
%s ' % (name, obj.book.title)
+ if name == "review":
+ return "%s of
%s " % (name, obj.book.title)
+ if name == "comment":
+ return "%s on
%s " % (name, obj.book.title)
+ if name == "quotation":
+ return "%s from
%s " % (name, obj.book.title)
return name
-@register.filter(name='next_shelf')
+
+@register.filter(name="next_shelf")
def get_next_shelf(current_shelf):
- ''' shelf you'd use to update reading progress '''
- if current_shelf == 'to-read':
- return 'reading'
- if current_shelf == 'reading':
- return 'read'
- if current_shelf == 'read':
- return 'read'
- return 'to-read'
+ """ shelf you'd use to update reading progress """
+ if current_shelf == "to-read":
+ return "reading"
+ if current_shelf == "reading":
+ return "read"
+ if current_shelf == "read":
+ return "read"
+ return "to-read"
+
@register.simple_tag(takes_context=False)
def related_status(notification):
- ''' for notifications '''
+ """ for notifications """
if not notification.related_status:
return None
- if hasattr(notification.related_status, 'quotation'):
+ if hasattr(notification.related_status, "quotation"):
return notification.related_status.quotation
- if hasattr(notification.related_status, 'review'):
+ if hasattr(notification.related_status, "review"):
return notification.related_status.review
- if hasattr(notification.related_status, 'comment'):
+ if hasattr(notification.related_status, "comment"):
return notification.related_status.comment
return notification.related_status
+
@register.simple_tag(takes_context=True)
def active_shelf(context, book):
- ''' check what shelf a user has a book on, if any '''
+ """ check what shelf a user has a book on, if any """
shelf = models.ShelfBook.objects.filter(
- shelf__user=context['request'].user,
- book__in=book.parent_work.editions.all()
+ shelf__user=context["request"].user, book__in=book.parent_work.editions.all()
).first()
- return shelf if shelf else {'book': book}
+ return shelf if shelf else {"book": book}
@register.simple_tag(takes_context=False)
def latest_read_through(book, user):
- ''' the most recent read activity '''
- return models.ReadThrough.objects.filter(
- user=user,
- book=book
- ).order_by('-start_date').first()
+ """ the most recent read activity """
+ return (
+ models.ReadThrough.objects.filter(user=user, book=book)
+ .order_by("-start_date")
+ .first()
+ )
@register.simple_tag(takes_context=False)
def active_read_through(book, user):
- ''' the most recent read activity '''
- return models.ReadThrough.objects.filter(
- user=user,
- book=book,
- finish_date__isnull=True
- ).order_by('-start_date').first()
+ """ the most recent read activity """
+ return (
+ models.ReadThrough.objects.filter(
+ user=user, book=book, finish_date__isnull=True
+ )
+ .order_by("-start_date")
+ .first()
+ )
@register.simple_tag(takes_context=False)
def comparison_bool(str1, str2):
- ''' idk why I need to write a tag for this, it reutrns a bool '''
+ """ idk why I need to write a tag for this, it reutrns a bool """
return str1 == str2
diff --git a/bookwyrm/tests/actions/__init__.py b/bookwyrm/tests/actions/__init__.py
deleted file mode 100644
index b6e690fd..00000000
--- a/bookwyrm/tests/actions/__init__.py
+++ /dev/null
@@ -1 +0,0 @@
-from . import *
diff --git a/bookwyrm/tests/activitypub/test_author.py b/bookwyrm/tests/activitypub/test_author.py
index fd31f105..e65f86b7 100644
--- a/bookwyrm/tests/activitypub/test_author.py
+++ b/bookwyrm/tests/activitypub/test_author.py
@@ -7,19 +7,18 @@ from bookwyrm import models
class Author(TestCase):
def setUp(self):
self.book = models.Edition.objects.create(
- title='Example Edition',
- remote_id='https://example.com/book/1',
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
)
self.author = models.Author.objects.create(
- name='Author fullname',
- aliases=['One', 'Two'],
- bio='bio bio bio',
+ name="Author fullname",
+ aliases=["One", "Two"],
+ bio="bio bio bio",
)
-
def test_serialize_model(self):
activity = self.author.to_activity()
- self.assertEqual(activity['id'], self.author.remote_id)
- self.assertIsInstance(activity['aliases'], list)
- self.assertEqual(activity['aliases'], ['One', 'Two'])
- self.assertEqual(activity['name'], 'Author fullname')
+ self.assertEqual(activity["id"], self.author.remote_id)
+ self.assertIsInstance(activity["aliases"], list)
+ self.assertEqual(activity["aliases"], ["One", "Two"])
+ self.assertEqual(activity["name"], "Author fullname")
diff --git a/bookwyrm/tests/activitypub/test_base_activity.py b/bookwyrm/tests/activitypub/test_base_activity.py
index d489fdaa..b3e28261 100644
--- a/bookwyrm/tests/activitypub/test_base_activity.py
+++ b/bookwyrm/tests/activitypub/test_base_activity.py
@@ -1,4 +1,4 @@
-''' tests the base functionality for activitypub dataclasses '''
+""" tests the base functionality for activitypub dataclasses """
from io import BytesIO
import json
import pathlib
@@ -10,228 +10,229 @@ from PIL import Image
import responses
from bookwyrm import activitypub
-from bookwyrm.activitypub.base_activity import ActivityObject, \
- resolve_remote_id, set_related_field
+from bookwyrm.activitypub.base_activity import (
+ ActivityObject,
+ resolve_remote_id,
+ set_related_field,
+)
from bookwyrm.activitypub import ActivitySerializerError
from bookwyrm import models
+
class BaseActivity(TestCase):
- ''' the super class for model-linked activitypub dataclasses '''
+ """ the super class for model-linked activitypub dataclasses """
+
def setUp(self):
- ''' we're probably going to re-use this so why copy/paste '''
+ """ we're probably going to re-use this so why copy/paste """
self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- self.user.remote_id = 'http://example.com/a/b'
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
+ self.user.remote_id = "http://example.com/a/b"
self.user.save(broadcast=False)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json'
- )
+ 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']
+ del self.userdata["icon"]
image_file = pathlib.Path(__file__).parent.joinpath(
- '../../static/images/default_avi.jpg')
+ "../../static/images/default_avi.jpg"
+ )
image = Image.open(image_file)
output = BytesIO()
image.save(output, format=image.format)
self.image_data = output.getvalue()
def test_init(self):
- ''' simple successfuly init '''
- instance = ActivityObject(id='a', type='b')
- self.assertTrue(hasattr(instance, 'id'))
- self.assertTrue(hasattr(instance, 'type'))
+ """ 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 '''
+ """ 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'))
+ """ 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 '''
+ """ replace an existing required field with a default field """
+
@dataclass(init=False)
class TestClass(ActivityObject):
- ''' test class with default field '''
- type: str = 'TestObject'
+ """ test class with default field """
- instance = TestClass(id='a')
- self.assertEqual(instance.id, 'a')
- self.assertEqual(instance.type, 'TestObject')
+ 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')
+ """ 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')
+ self.assertEqual(serialized["id"], "a")
+ self.assertEqual(serialized["type"], "b")
@responses.activate
def test_resolve_remote_id(self):
- ''' look up or load remote data '''
+ """ look up or load remote data """
# existing item
- result = resolve_remote_id('http://example.com/a/b', model=models.User)
+ result = resolve_remote_id("http://example.com/a/b", model=models.User)
self.assertEqual(result, self.user)
# remote item
responses.add(
responses.GET,
- 'https://example.com/user/mouse',
+ "https://example.com/user/mouse",
json=self.userdata,
- status=200)
+ status=200,
+ )
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
result = resolve_remote_id(
- 'https://example.com/user/mouse', model=models.User)
+ "https://example.com/user/mouse", model=models.User
+ )
self.assertIsInstance(result, models.User)
- self.assertEqual(result.remote_id, 'https://example.com/user/mouse')
- self.assertEqual(result.name, 'MOUSE?? MOUSE!!')
+ self.assertEqual(result.remote_id, "https://example.com/user/mouse")
+ self.assertEqual(result.name, "MOUSE?? MOUSE!!")
def test_to_model_invalid_model(self):
- ''' catch mismatch between activity type and model type '''
- instance = ActivityObject(id='a', type='b')
+ """ catch mismatch between activity type and model type """
+ instance = ActivityObject(id="a", type="b")
with self.assertRaises(ActivitySerializerError):
instance.to_model(model=models.User)
-
@responses.activate
def test_to_model_image(self):
- ''' update an image field '''
+ """ update an image field """
activity = activitypub.Person(
id=self.user.remote_id,
- name='New Name',
- preferredUsername='mouse',
- inbox='http://www.com/',
- outbox='http://www.com/',
- followers='',
- summary='',
- publicKey={
- 'id': 'hi',
- 'owner': self.user.remote_id,
- 'publicKeyPem': 'hi'},
+ name="New Name",
+ preferredUsername="mouse",
+ inbox="http://www.com/",
+ outbox="http://www.com/",
+ followers="",
+ summary="",
+ publicKey={"id": "hi", "owner": self.user.remote_id, "publicKeyPem": "hi"},
endpoints={},
- icon={
- 'type': 'Image',
- 'url': 'http://www.example.com/image.jpg'
- }
+ icon={"type": "Image", "url": "http://www.example.com/image.jpg"},
)
responses.add(
responses.GET,
- 'http://www.example.com/image.jpg',
+ "http://www.example.com/image.jpg",
body=self.image_data,
- status=200)
+ status=200,
+ )
self.assertIsNone(self.user.avatar.name)
with self.assertRaises(ValueError):
- self.user.avatar.file #pylint: disable=pointless-statement
+ self.user.avatar.file # pylint: disable=pointless-statement
# this would trigger a broadcast because it's a local user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
activity.to_model(model=models.User, instance=self.user)
self.assertIsNotNone(self.user.avatar.file)
- self.assertEqual(self.user.name, 'New Name')
- self.assertEqual(self.user.key_pair.public_key, 'hi')
+ self.assertEqual(self.user.name, "New Name")
+ self.assertEqual(self.user.key_pair.public_key, "hi")
def test_to_model_many_to_many(self):
- ''' annoying that these all need special handling '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ annoying that these all need special handling """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
status = models.Status.objects.create(
- content='test status',
+ content="test status",
user=self.user,
)
book = models.Edition.objects.create(
- title='Test Edition', remote_id='http://book.com/book')
+ title="Test Edition", remote_id="http://book.com/book"
+ )
update_data = activitypub.Note(
id=status.remote_id,
content=status.content,
attributedTo=self.user.remote_id,
- published='hi',
+ published="hi",
to=[],
cc=[],
tag=[
+ {"type": "Mention", "name": "gerald", "href": "http://example.com/a/b"},
{
- 'type': 'Mention',
- 'name': 'gerald',
- 'href': 'http://example.com/a/b'
+ "type": "Edition",
+ "name": "gerald j. books",
+ "href": "http://book.com/book",
},
- {
- 'type': 'Edition',
- 'name': 'gerald j. books',
- 'href': 'http://book.com/book'
- },
- ]
+ ],
)
update_data.to_model(model=models.Status, instance=status)
self.assertEqual(status.mention_users.first(), self.user)
self.assertEqual(status.mention_books.first(), book)
-
@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 '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """these are reversed relationships, where the secondary object
+ keys the primary object but not vice versa"""
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
status = models.Status.objects.create(
- content='test status',
+ content="test status",
user=self.user,
)
update_data = activitypub.Note(
id=status.remote_id,
content=status.content,
attributedTo=self.user.remote_id,
- published='hi',
+ published="hi",
to=[],
cc=[],
- attachment=[{
- 'url': 'http://www.example.com/image.jpg',
- 'name': 'alt text',
- 'type': 'Image',
- }],
+ attachment=[
+ {
+ "url": "http://www.example.com/image.jpg",
+ "name": "alt text",
+ "type": "Image",
+ }
+ ],
)
responses.add(
responses.GET,
- 'http://www.example.com/image.jpg',
+ "http://www.example.com/image.jpg",
body=self.image_data,
- status=200)
+ status=200,
+ )
# sets the celery task call to the function call
- with patch(
- 'bookwyrm.activitypub.base_activity.set_related_field.delay'):
- update_data.to_model(model=models.Status, instance=status)
+ with patch("bookwyrm.activitypub.base_activity.set_related_field.delay"):
+ with patch("bookwyrm.models.status.Status.ignore_activity") as discarder:
+ discarder.return_value = False
+ update_data.to_model(model=models.Status, instance=status)
self.assertIsNone(status.attachments.first())
-
@responses.activate
def test_set_related_field(self):
- ''' celery task to add back-references to created objects '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ celery task to add back-references to created objects """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
status = models.Status.objects.create(
- content='test status',
+ content="test status",
user=self.user,
)
data = {
- 'url': 'http://www.example.com/image.jpg',
- 'name': 'alt text',
- 'type': 'Image',
+ "url": "http://www.example.com/image.jpg",
+ "name": "alt text",
+ "type": "Image",
}
responses.add(
responses.GET,
- 'http://www.example.com/image.jpg',
+ "http://www.example.com/image.jpg",
body=self.image_data,
- status=200)
- set_related_field(
- 'Image', 'Status', 'status', status.remote_id, data)
+ status=200,
+ )
+ set_related_field("Image", "Status", "status", status.remote_id, data)
self.assertIsInstance(status.attachments.first(), models.Image)
self.assertIsNotNone(status.attachments.first().image)
diff --git a/bookwyrm/tests/activitypub/test_person.py b/bookwyrm/tests/activitypub/test_person.py
index 06240281..67aaf891 100644
--- a/bookwyrm/tests/activitypub/test_person.py
+++ b/bookwyrm/tests/activitypub/test_person.py
@@ -9,23 +9,19 @@ from bookwyrm import activitypub, models
class Person(TestCase):
def setUp(self):
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json'
- )
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
self.user_data = json.loads(datafile.read_bytes())
-
def test_load_user_data(self):
activity = activitypub.Person(**self.user_data)
- self.assertEqual(activity.id, 'https://example.com/user/mouse')
- self.assertEqual(activity.preferredUsername, 'mouse')
- self.assertEqual(activity.type, 'Person')
-
+ self.assertEqual(activity.id, "https://example.com/user/mouse")
+ self.assertEqual(activity.preferredUsername, "mouse")
+ self.assertEqual(activity.type, "Person")
def test_user_to_model(self):
activity = activitypub.Person(**self.user_data)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
user = activity.to_model(model=models.User)
- self.assertEqual(user.username, 'mouse@example.com')
- self.assertEqual(user.remote_id, 'https://example.com/user/mouse')
+ self.assertEqual(user.username, "mouse@example.com")
+ self.assertEqual(user.remote_id, "https://example.com/user/mouse")
self.assertFalse(user.local)
diff --git a/bookwyrm/tests/activitypub/test_quotation.py b/bookwyrm/tests/activitypub/test_quotation.py
index 1cd1f05d..1f429dd2 100644
--- a/bookwyrm/tests/activitypub/test_quotation.py
+++ b/bookwyrm/tests/activitypub/test_quotation.py
@@ -1,4 +1,4 @@
-''' quotation activty object serializer class '''
+""" quotation activty object serializer class """
import json
import pathlib
from unittest.mock import patch
@@ -8,43 +8,40 @@ from bookwyrm import activitypub, models
class Quotation(TestCase):
- ''' we have hecka ways to create statuses '''
+ """ we have hecka ways to create statuses """
+
def setUp(self):
- ''' model objects we'll need '''
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ """ model objects we'll need """
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
+ "mouse",
+ "mouse@mouse.mouse",
+ "mouseword",
local=False,
- inbox='https://example.com/user/mouse/inbox',
- outbox='https://example.com/user/mouse/outbox',
- remote_id='https://example.com/user/mouse',
+ inbox="https://example.com/user/mouse/inbox",
+ outbox="https://example.com/user/mouse/outbox",
+ remote_id="https://example.com/user/mouse",
)
self.book = models.Edition.objects.create(
- title='Example Edition',
- remote_id='https://example.com/book/1',
- )
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_quotation.json'
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
)
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_quotation.json")
self.status_data = json.loads(datafile.read_bytes())
-
def test_quotation_activity(self):
- ''' create a Quoteation ap object from json '''
+ """ create a Quoteation ap object from json """
quotation = activitypub.Quotation(**self.status_data)
- self.assertEqual(quotation.type, 'Quotation')
- self.assertEqual(
- quotation.id, 'https://example.com/user/mouse/quotation/13')
- self.assertEqual(quotation.content, 'commentary')
- self.assertEqual(quotation.quote, 'quote body')
- self.assertEqual(quotation.inReplyToBook, 'https://example.com/book/1')
- self.assertEqual(
- quotation.published, '2020-05-10T02:38:31.150343+00:00')
-
+ self.assertEqual(quotation.type, "Quotation")
+ self.assertEqual(quotation.id, "https://example.com/user/mouse/quotation/13")
+ self.assertEqual(quotation.content, "commentary")
+ self.assertEqual(quotation.quote, "quote body")
+ self.assertEqual(quotation.inReplyToBook, "https://example.com/book/1")
+ self.assertEqual(quotation.published, "2020-05-10T02:38:31.150343+00:00")
def test_activity_to_model(self):
- ''' create a model instance from an activity object '''
+ """ create a model instance from an activity object """
activity = activitypub.Quotation(**self.status_data)
quotation = activity.to_model(model=models.Quotation)
diff --git a/bookwyrm/tests/connectors/test_abstract_connector.py b/bookwyrm/tests/connectors/test_abstract_connector.py
index 6e912858..9aa78f6a 100644
--- a/bookwyrm/tests/connectors/test_abstract_connector.py
+++ b/bookwyrm/tests/connectors/test_abstract_connector.py
@@ -1,4 +1,4 @@
-''' testing book data connectors '''
+""" testing book data connectors """
from unittest.mock import patch
from django.test import TestCase
import responses
@@ -10,105 +10,115 @@ from bookwyrm.settings import DOMAIN
class AbstractConnector(TestCase):
- ''' generic code for connecting to outside data sources '''
+ """ generic code for connecting to outside data sources """
+
def setUp(self):
- ''' we need an example connector '''
+ """ we need an example connector """
self.connector_info = models.Connector.objects.create(
- identifier='example.com',
- connector_file='openlibrary',
- base_url='https://example.com',
- books_url='https://example.com/books',
- covers_url='https://example.com/covers',
- search_url='https://example.com/search?q=',
+ identifier="example.com",
+ connector_file="openlibrary",
+ base_url="https://example.com",
+ books_url="https://example.com/books",
+ covers_url="https://example.com/covers",
+ search_url="https://example.com/search?q=",
)
work_data = {
- 'id': 'abc1',
- 'title': 'Test work',
- 'type': 'work',
- 'openlibraryKey': 'OL1234W',
+ "id": "abc1",
+ "title": "Test work",
+ "type": "work",
+ "openlibraryKey": "OL1234W",
}
self.work_data = work_data
edition_data = {
- 'id': 'abc2',
- 'title': 'Test edition',
- 'type': 'edition',
- 'openlibraryKey': 'OL1234M',
+ "id": "abc2",
+ "title": "Test edition",
+ "type": "edition",
+ "openlibraryKey": "OL1234M",
}
self.edition_data = edition_data
class TestConnector(abstract_connector.AbstractConnector):
- ''' nothing added here '''
+ """ nothing added here """
+
def format_search_result(self, search_result):
return search_result
+
def parse_search_data(self, data):
return data
+
+ def format_isbn_search_result(self, search_result):
+ return search_result
+
+ def parse_isbn_search_data(self, data):
+ return data
+
def is_work_data(self, data):
- return data['type'] == 'work'
+ return data["type"] == "work"
+
def get_edition_from_work_data(self, data):
return edition_data
+
def get_work_from_edition_data(self, data):
return work_data
+
def get_authors_from_data(self, data):
return []
+
def expand_book_data(self, book):
pass
- self.connector = TestConnector('example.com')
+
+ self.connector = TestConnector("example.com")
self.connector.book_mappings = [
- Mapping('id'),
- Mapping('title'),
- Mapping('openlibraryKey'),
+ Mapping("id"),
+ Mapping("title"),
+ Mapping("openlibraryKey"),
]
self.book = models.Edition.objects.create(
- title='Test Book', remote_id='https://example.com/book/1234',
- openlibrary_key='OL1234M')
-
+ title="Test Book",
+ remote_id="https://example.com/book/1234",
+ openlibrary_key="OL1234M",
+ )
def test_abstract_connector_init(self):
- ''' barebones connector for search with defaults '''
+ """ barebones connector for search with defaults """
self.assertIsInstance(self.connector.book_mappings, list)
-
def test_is_available(self):
- ''' this isn't used.... '''
+ """ this isn't used.... """
self.assertTrue(self.connector.is_available())
self.connector.max_query_count = 1
self.connector.connector.query_count = 2
self.assertFalse(self.connector.is_available())
-
def test_get_or_create_book_existing(self):
- ''' find an existing book by remote/origin id '''
+ """ find an existing book by remote/origin id """
self.assertEqual(models.Book.objects.count(), 1)
self.assertEqual(
- self.book.remote_id, 'https://%s/book/%d' % (DOMAIN, self.book.id))
- self.assertEqual(
- self.book.origin_id, 'https://example.com/book/1234')
+ self.book.remote_id, "https://%s/book/%d" % (DOMAIN, self.book.id)
+ )
+ self.assertEqual(self.book.origin_id, "https://example.com/book/1234")
# dedupe by origin id
- result = self.connector.get_or_create_book(
- 'https://example.com/book/1234')
+ result = self.connector.get_or_create_book("https://example.com/book/1234")
self.assertEqual(models.Book.objects.count(), 1)
self.assertEqual(result, self.book)
# dedupe by remote id
result = self.connector.get_or_create_book(
- 'https://%s/book/%d' % (DOMAIN, self.book.id))
+ "https://%s/book/%d" % (DOMAIN, self.book.id)
+ )
self.assertEqual(models.Book.objects.count(), 1)
self.assertEqual(result, self.book)
@responses.activate
def test_get_or_create_book_deduped(self):
- ''' load remote data and deduplicate '''
+ """ load remote data and deduplicate """
responses.add(
- responses.GET,
- 'https://example.com/book/abcd',
- json=self.edition_data
+ responses.GET, "https://example.com/book/abcd", json=self.edition_data
)
- with patch(
- 'bookwyrm.connectors.abstract_connector.load_more_data.delay'):
- result = self.connector.get_or_create_book(
- 'https://example.com/book/abcd')
+ with patch("bookwyrm.connectors.abstract_connector.load_more_data.delay"):
+ result = self.connector.get_or_create_book("https://example.com/book/abcd")
self.assertEqual(result, self.book)
self.assertEqual(models.Edition.objects.count(), 1)
self.assertEqual(models.Edition.objects.count(), 1)
diff --git a/bookwyrm/tests/connectors/test_abstract_minimal_connector.py b/bookwyrm/tests/connectors/test_abstract_minimal_connector.py
index 0c6d2535..957d3233 100644
--- a/bookwyrm/tests/connectors/test_abstract_minimal_connector.py
+++ b/bookwyrm/tests/connectors/test_abstract_minimal_connector.py
@@ -1,4 +1,4 @@
-''' testing book data connectors '''
+""" testing book data connectors """
from django.test import TestCase
import responses
@@ -8,93 +8,101 @@ from bookwyrm.connectors.abstract_connector import Mapping, SearchResult
class AbstractConnector(TestCase):
- ''' generic code for connecting to outside data sources '''
+ """ generic code for connecting to outside data sources """
+
def setUp(self):
- ''' we need an example connector '''
+ """ we need an example connector """
self.connector_info = models.Connector.objects.create(
- identifier='example.com',
- connector_file='openlibrary',
- base_url='https://example.com',
- books_url='https://example.com/books',
- covers_url='https://example.com/covers',
- search_url='https://example.com/search?q=',
+ identifier="example.com",
+ connector_file="openlibrary",
+ base_url="https://example.com",
+ books_url="https://example.com/books",
+ covers_url="https://example.com/covers",
+ search_url="https://example.com/search?q=",
+ isbn_search_url="https://example.com/isbn",
)
class TestConnector(abstract_connector.AbstractMinimalConnector):
- ''' nothing added here '''
+ """ nothing added here """
+
def format_search_result(self, search_result):
return search_result
+
def get_or_create_book(self, remote_id):
pass
+
def parse_search_data(self, data):
return data
- self.test_connector = TestConnector('example.com')
+ def format_isbn_search_result(self, search_result):
+ return search_result
+
+ def parse_isbn_search_data(self, data):
+ return data
+
+ self.test_connector = TestConnector("example.com")
def test_abstract_minimal_connector_init(self):
- ''' barebones connector for search with defaults '''
+ """ barebones connector for search with defaults """
connector = self.test_connector
self.assertEqual(connector.connector, self.connector_info)
- self.assertEqual(connector.base_url, 'https://example.com')
- self.assertEqual(connector.books_url, 'https://example.com/books')
- self.assertEqual(connector.covers_url, 'https://example.com/covers')
- self.assertEqual(connector.search_url, 'https://example.com/search?q=')
+ self.assertEqual(connector.base_url, "https://example.com")
+ self.assertEqual(connector.books_url, "https://example.com/books")
+ self.assertEqual(connector.covers_url, "https://example.com/covers")
+ self.assertEqual(connector.search_url, "https://example.com/search?q=")
+ self.assertEqual(connector.isbn_search_url, "https://example.com/isbn")
self.assertIsNone(connector.name)
- self.assertEqual(connector.identifier, 'example.com')
+ self.assertEqual(connector.identifier, "example.com")
self.assertIsNone(connector.max_query_count)
self.assertFalse(connector.local)
-
@responses.activate
def test_search(self):
- ''' makes an http request to the outside service '''
+ """ makes an http request to the outside service """
responses.add(
responses.GET,
- 'https://example.com/search?q=a%20book%20title',
- json=['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l'],
- status=200)
- results = self.test_connector.search('a book title')
+ "https://example.com/search?q=a%20book%20title",
+ json=["a", "b", "c", "d", "e", "f", "g", "h", "i", "j", "k", "l"],
+ status=200,
+ )
+ results = self.test_connector.search("a book title")
self.assertEqual(len(results), 10)
- self.assertEqual(results[0], 'a')
- self.assertEqual(results[1], 'b')
- self.assertEqual(results[2], 'c')
-
+ self.assertEqual(results[0], "a")
+ self.assertEqual(results[1], "b")
+ self.assertEqual(results[2], "c")
def test_search_result(self):
- ''' a class that stores info about a search result '''
+ """ a class that stores info about a search result """
result = SearchResult(
- title='Title',
- key='https://example.com/book/1',
- author='Author Name',
- year='1850',
+ title="Title",
+ key="https://example.com/book/1",
+ author="Author Name",
+ year="1850",
connector=self.test_connector,
)
# there's really not much to test here, it's just a dataclass
self.assertEqual(result.confidence, 1)
- self.assertEqual(result.title, 'Title')
-
+ self.assertEqual(result.title, "Title")
def test_create_mapping(self):
- ''' maps remote fields for book data to bookwyrm activitypub fields '''
- mapping = Mapping('isbn')
- self.assertEqual(mapping.local_field, 'isbn')
- self.assertEqual(mapping.remote_field, 'isbn')
- self.assertEqual(mapping.formatter('bb'), 'bb')
-
+ """ maps remote fields for book data to bookwyrm activitypub fields """
+ mapping = Mapping("isbn")
+ self.assertEqual(mapping.local_field, "isbn")
+ self.assertEqual(mapping.remote_field, "isbn")
+ self.assertEqual(mapping.formatter("bb"), "bb")
def test_create_mapping_with_remote(self):
- ''' the remote field is different than the local field '''
- mapping = Mapping('isbn', remote_field='isbn13')
- self.assertEqual(mapping.local_field, 'isbn')
- self.assertEqual(mapping.remote_field, 'isbn13')
- self.assertEqual(mapping.formatter('bb'), 'bb')
-
+ """ the remote field is different than the local field """
+ mapping = Mapping("isbn", remote_field="isbn13")
+ self.assertEqual(mapping.local_field, "isbn")
+ self.assertEqual(mapping.remote_field, "isbn13")
+ self.assertEqual(mapping.formatter("bb"), "bb")
def test_create_mapping_with_formatter(self):
- ''' a function is provided to modify the data '''
- formatter = lambda x: 'aa' + x
- mapping = Mapping('isbn', formatter=formatter)
- self.assertEqual(mapping.local_field, 'isbn')
- self.assertEqual(mapping.remote_field, 'isbn')
+ """ a function is provided to modify the data """
+ formatter = lambda x: "aa" + x
+ mapping = Mapping("isbn", formatter=formatter)
+ self.assertEqual(mapping.local_field, "isbn")
+ self.assertEqual(mapping.remote_field, "isbn")
self.assertEqual(mapping.formatter, formatter)
- self.assertEqual(mapping.formatter('bb'), 'aabb')
+ self.assertEqual(mapping.formatter("bb"), "aabb")
diff --git a/bookwyrm/tests/connectors/test_bookwyrm_connector.py b/bookwyrm/tests/connectors/test_bookwyrm_connector.py
index 6b00b0e3..1fc71688 100644
--- a/bookwyrm/tests/connectors/test_bookwyrm_connector.py
+++ b/bookwyrm/tests/connectors/test_bookwyrm_connector.py
@@ -1,4 +1,4 @@
-''' testing book data connectors '''
+""" testing book data connectors """
import json
import pathlib
from django.test import TestCase
@@ -9,39 +9,36 @@ from bookwyrm.connectors.abstract_connector import SearchResult
class BookWyrmConnector(TestCase):
- ''' this connector doesn't do much, just search '''
- def setUp(self):
- ''' create the connector '''
- models.Connector.objects.create(
- identifier='example.com',
- connector_file='bookwyrm_connector',
- base_url='https://example.com',
- books_url='https://example.com',
- covers_url='https://example.com/images/covers',
- search_url='https://example.com/search?q=',
- )
- self.connector = Connector('example.com')
+ """ this connector doesn't do much, just search """
- work_file = pathlib.Path(__file__).parent.joinpath(
- '../data/bw_work.json')
- edition_file = pathlib.Path(__file__).parent.joinpath(
- '../data/bw_edition.json')
+ def setUp(self):
+ """ create the connector """
+ models.Connector.objects.create(
+ identifier="example.com",
+ connector_file="bookwyrm_connector",
+ base_url="https://example.com",
+ books_url="https://example.com",
+ covers_url="https://example.com/images/covers",
+ search_url="https://example.com/search?q=",
+ )
+ self.connector = Connector("example.com")
+
+ work_file = pathlib.Path(__file__).parent.joinpath("../data/bw_work.json")
+ edition_file = pathlib.Path(__file__).parent.joinpath("../data/bw_edition.json")
self.work_data = json.loads(work_file.read_bytes())
self.edition_data = json.loads(edition_file.read_bytes())
-
def test_format_search_result(self):
- ''' create a SearchResult object from search response json '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/bw_search.json')
+ """ create a SearchResult object from search response json """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/bw_search.json")
search_data = json.loads(datafile.read_bytes())
results = self.connector.parse_search_data(search_data)
self.assertIsInstance(results, list)
result = self.connector.format_search_result(results[0])
self.assertIsInstance(result, SearchResult)
- self.assertEqual(result.title, 'Jonathan Strange and Mr Norrell')
- self.assertEqual(result.key, 'https://example.com/book/122')
- self.assertEqual(result.author, 'Susanna Clarke')
+ self.assertEqual(result.title, "Jonathan Strange and Mr Norrell")
+ self.assertEqual(result.key, "https://example.com/book/122")
+ self.assertEqual(result.author, "Susanna Clarke")
self.assertEqual(result.year, 2017)
self.assertEqual(result.connector, self.connector)
diff --git a/bookwyrm/tests/connectors/test_connector_manager.py b/bookwyrm/tests/connectors/test_connector_manager.py
index 783b5a27..4410e011 100644
--- a/bookwyrm/tests/connectors/test_connector_manager.py
+++ b/bookwyrm/tests/connectors/test_connector_manager.py
@@ -1,54 +1,49 @@
-''' interface between the app and various connectors '''
+""" interface between the app and various connectors """
from django.test import TestCase
from bookwyrm import models
from bookwyrm.connectors import connector_manager
-from bookwyrm.connectors.bookwyrm_connector \
- import Connector as BookWyrmConnector
-from bookwyrm.connectors.self_connector \
- import Connector as SelfConnector
+from bookwyrm.connectors.bookwyrm_connector import Connector as BookWyrmConnector
+from bookwyrm.connectors.self_connector import Connector as SelfConnector
class ConnectorManager(TestCase):
- ''' interface between the app and various connectors '''
+ """ interface between the app and various connectors """
+
def setUp(self):
- ''' we'll need some books and a connector info entry '''
- self.work = models.Work.objects.create(
- title='Example Work'
- )
+ """ we'll need some books and a connector info entry """
+ self.work = models.Work.objects.create(title="Example Work")
self.edition = models.Edition.objects.create(
- title='Example Edition',
- parent_work=self.work
+ title="Example Edition", parent_work=self.work
)
self.work.default_edition = self.edition
self.work.save()
self.connector = models.Connector.objects.create(
- identifier='test_connector',
+ identifier="test_connector",
priority=1,
local=True,
- connector_file='self_connector',
- base_url='http://test.com/',
- books_url='http://test.com/',
- covers_url='http://test.com/',
+ connector_file="self_connector",
+ base_url="http://test.com/",
+ books_url="http://test.com/",
+ covers_url="http://test.com/",
)
-
def test_get_or_create_connector(self):
- ''' loads a connector if the data source is known or creates one '''
- remote_id = 'https://example.com/object/1'
+ """ loads a connector if the data source is known or creates one """
+ remote_id = "https://example.com/object/1"
connector = connector_manager.get_or_create_connector(remote_id)
self.assertIsInstance(connector, BookWyrmConnector)
- self.assertEqual(connector.identifier, 'example.com')
- self.assertEqual(connector.base_url, 'https://example.com')
+ self.assertEqual(connector.identifier, "example.com")
+ self.assertEqual(connector.base_url, "https://example.com")
same_connector = connector_manager.get_or_create_connector(remote_id)
self.assertEqual(connector.identifier, same_connector.identifier)
def test_get_connectors(self):
- ''' load all connectors '''
- remote_id = 'https://example.com/object/1'
+ """ load all connectors """
+ remote_id = "https://example.com/object/1"
connector_manager.get_or_create_connector(remote_id)
connectors = list(connector_manager.get_connectors())
self.assertEqual(len(connectors), 2)
@@ -56,28 +51,28 @@ class ConnectorManager(TestCase):
self.assertIsInstance(connectors[1], BookWyrmConnector)
def test_search(self):
- ''' search all connectors '''
- results = connector_manager.search('Example')
+ """ search all connectors """
+ results = connector_manager.search("Example")
self.assertEqual(len(results), 1)
- self.assertIsInstance(results[0]['connector'], SelfConnector)
- self.assertEqual(len(results[0]['results']), 1)
- self.assertEqual(results[0]['results'][0].title, 'Example Edition')
+ self.assertIsInstance(results[0]["connector"], SelfConnector)
+ self.assertEqual(len(results[0]["results"]), 1)
+ self.assertEqual(results[0]["results"][0].title, "Example Edition")
def test_local_search(self):
- ''' search only the local database '''
- results = connector_manager.local_search('Example')
+ """ search only the local database """
+ results = connector_manager.local_search("Example")
self.assertEqual(len(results), 1)
- self.assertEqual(results[0].title, 'Example Edition')
+ self.assertEqual(results[0].title, "Example Edition")
def test_first_search_result(self):
- ''' only get one search result '''
- result = connector_manager.first_search_result('Example')
- self.assertEqual(result.title, 'Example Edition')
- no_result = connector_manager.first_search_result('dkjfhg')
+ """ only get one search result """
+ result = connector_manager.first_search_result("Example")
+ self.assertEqual(result.title, "Example Edition")
+ no_result = connector_manager.first_search_result("dkjfhg")
self.assertIsNone(no_result)
def test_load_connector(self):
- ''' load a connector object from the database entry '''
+ """ load a connector object from the database entry """
connector = connector_manager.load_connector(self.connector)
self.assertIsInstance(connector, SelfConnector)
- self.assertEqual(connector.identifier, 'test_connector')
+ self.assertEqual(connector.identifier, "test_connector")
diff --git a/bookwyrm/tests/connectors/test_openlibrary_connector.py b/bookwyrm/tests/connectors/test_openlibrary_connector.py
index 576e353b..bb018830 100644
--- a/bookwyrm/tests/connectors/test_openlibrary_connector.py
+++ b/bookwyrm/tests/connectors/test_openlibrary_connector.py
@@ -1,4 +1,4 @@
-''' testing book data connectors '''
+""" testing book data connectors """
import json
import pathlib
from unittest.mock import patch
@@ -9,224 +9,231 @@ import responses
from bookwyrm import models
from bookwyrm.connectors.openlibrary import Connector
from bookwyrm.connectors.openlibrary import get_languages, get_description
-from bookwyrm.connectors.openlibrary import pick_default_edition, \
- get_openlibrary_key
+from bookwyrm.connectors.openlibrary import pick_default_edition, get_openlibrary_key
from bookwyrm.connectors.abstract_connector import SearchResult
from bookwyrm.connectors.connector_manager import ConnectorException
class Openlibrary(TestCase):
- ''' test loading data from openlibrary.org '''
- def setUp(self):
- ''' creates the connector we'll use '''
- models.Connector.objects.create(
- identifier='openlibrary.org',
- name='OpenLibrary',
- connector_file='openlibrary',
- base_url='https://openlibrary.org',
- books_url='https://openlibrary.org',
- covers_url='https://covers.openlibrary.org',
- search_url='https://openlibrary.org/search?q=',
- )
- self.connector = Connector('openlibrary.org')
+ """ test loading data from openlibrary.org """
- work_file = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_work.json')
- edition_file = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_edition.json')
+ def setUp(self):
+ """ creates the connector we'll use """
+ models.Connector.objects.create(
+ identifier="openlibrary.org",
+ name="OpenLibrary",
+ connector_file="openlibrary",
+ base_url="https://openlibrary.org",
+ books_url="https://openlibrary.org",
+ covers_url="https://covers.openlibrary.org",
+ search_url="https://openlibrary.org/search?q=",
+ isbn_search_url="https://openlibrary.org/isbn",
+ )
+ self.connector = Connector("openlibrary.org")
+
+ work_file = pathlib.Path(__file__).parent.joinpath("../data/ol_work.json")
+ edition_file = pathlib.Path(__file__).parent.joinpath("../data/ol_edition.json")
edition_list_file = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_edition_list.json')
+ "../data/ol_edition_list.json"
+ )
self.work_data = json.loads(work_file.read_bytes())
self.edition_data = json.loads(edition_file.read_bytes())
self.edition_list_data = json.loads(edition_list_file.read_bytes())
-
def test_get_remote_id_from_data(self):
- ''' format the remote id from the data '''
- data = {'key': '/work/OL1234W'}
+ """ format the remote id from the data """
+ data = {"key": "/work/OL1234W"}
result = self.connector.get_remote_id_from_data(data)
- self.assertEqual(result, 'https://openlibrary.org/work/OL1234W')
+ self.assertEqual(result, "https://openlibrary.org/work/OL1234W")
# error handlding
with self.assertRaises(ConnectorException):
self.connector.get_remote_id_from_data({})
-
def test_is_work_data(self):
- ''' detect if the loaded json is a work '''
+ """ detect if the loaded json is a work """
self.assertEqual(self.connector.is_work_data(self.work_data), True)
self.assertEqual(self.connector.is_work_data(self.edition_data), False)
-
@responses.activate
def test_get_edition_from_work_data(self):
- ''' loads a list of editions '''
- data = {'key': '/work/OL1234W'}
+ """ loads a list of editions """
+ data = {"key": "/work/OL1234W"}
responses.add(
responses.GET,
- 'https://openlibrary.org/work/OL1234W/editions',
- json={'entries': []},
- status=200)
- with patch('bookwyrm.connectors.openlibrary.pick_default_edition') \
- as pick_edition:
- pick_edition.return_value = 'hi'
+ "https://openlibrary.org/work/OL1234W/editions",
+ json={"entries": []},
+ status=200,
+ )
+ with patch(
+ "bookwyrm.connectors.openlibrary.pick_default_edition"
+ ) as pick_edition:
+ pick_edition.return_value = "hi"
result = self.connector.get_edition_from_work_data(data)
- self.assertEqual(result, 'hi')
-
+ self.assertEqual(result, "hi")
@responses.activate
def test_get_work_from_edition_data(self):
- ''' loads a list of editions '''
- data = {'works': [{'key': '/work/OL1234W'}]}
+ """ loads a list of editions """
+ data = {"works": [{"key": "/work/OL1234W"}]}
responses.add(
responses.GET,
- 'https://openlibrary.org/work/OL1234W',
- json={'hi': 'there'},
- status=200)
+ "https://openlibrary.org/work/OL1234W",
+ json={"hi": "there"},
+ status=200,
+ )
result = self.connector.get_work_from_edition_data(data)
- self.assertEqual(result, {'hi': 'there'})
-
+ self.assertEqual(result, {"hi": "there"})
@responses.activate
def test_get_authors_from_data(self):
- ''' find authors in data '''
+ """ find authors in data """
responses.add(
responses.GET,
- 'https://openlibrary.org/authors/OL382982A',
+ "https://openlibrary.org/authors/OL382982A",
json={
"name": "George Elliott",
"personal_name": "George Elliott",
"last_modified": {
"type": "/type/datetime",
- "value": "2008-08-31 10:09:33.413686"
- },
+ "value": "2008-08-31 10:09:33.413686",
+ },
"key": "/authors/OL453734A",
- "type": {
- "key": "/type/author"
- },
+ "type": {"key": "/type/author"},
"id": 1259965,
- "revision": 2
+ "revision": 2,
},
- status=200)
+ status=200,
+ )
results = self.connector.get_authors_from_data(self.work_data)
result = list(results)[0]
self.assertIsInstance(result, models.Author)
- self.assertEqual(result.name, 'George Elliott')
- self.assertEqual(result.openlibrary_key, 'OL453734A')
-
+ self.assertEqual(result.name, "George Elliott")
+ self.assertEqual(result.openlibrary_key, "OL453734A")
def test_get_cover_url(self):
- ''' formats a url that should contain the cover image '''
- blob = ['image']
+ """ formats a url that should contain the cover image """
+ blob = ["image"]
result = self.connector.get_cover_url(blob)
- self.assertEqual(
- result, 'https://covers.openlibrary.org/b/id/image-L.jpg')
+ self.assertEqual(result, "https://covers.openlibrary.org/b/id/image-L.jpg")
def test_parse_search_result(self):
- ''' extract the results from the search json response '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_search.json')
+ """ extract the results from the search json response """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ol_search.json")
search_data = json.loads(datafile.read_bytes())
result = self.connector.parse_search_data(search_data)
self.assertIsInstance(result, list)
self.assertEqual(len(result), 2)
-
def test_format_search_result(self):
- ''' translate json from openlibrary into SearchResult '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_search.json')
+ """ translate json from openlibrary into SearchResult """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ol_search.json")
search_data = json.loads(datafile.read_bytes())
results = self.connector.parse_search_data(search_data)
self.assertIsInstance(results, list)
result = self.connector.format_search_result(results[0])
self.assertIsInstance(result, SearchResult)
- self.assertEqual(result.title, 'This Is How You Lose the Time War')
- self.assertEqual(
- result.key, 'https://openlibrary.org/works/OL20639540W')
- self.assertEqual(result.author, 'Amal El-Mohtar, Max Gladstone')
+ self.assertEqual(result.title, "This Is How You Lose the Time War")
+ self.assertEqual(result.key, "https://openlibrary.org/works/OL20639540W")
+ self.assertEqual(result.author, "Amal El-Mohtar, Max Gladstone")
self.assertEqual(result.year, 2019)
self.assertEqual(result.connector, self.connector)
+ def test_parse_isbn_search_result(self):
+ """ extract the results from the search json response """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ol_isbn_search.json")
+ search_data = json.loads(datafile.read_bytes())
+ result = self.connector.parse_isbn_search_data(search_data)
+ self.assertIsInstance(result, list)
+ self.assertEqual(len(result), 1)
+
+ def test_format_isbn_search_result(self):
+ """ translate json from openlibrary into SearchResult """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ol_isbn_search.json")
+ search_data = json.loads(datafile.read_bytes())
+ results = self.connector.parse_isbn_search_data(search_data)
+ self.assertIsInstance(results, list)
+
+ result = self.connector.format_isbn_search_result(results[0])
+ self.assertIsInstance(result, SearchResult)
+ self.assertEqual(result.title, "Les ombres errantes")
+ self.assertEqual(result.key, "https://openlibrary.org/books/OL16262504M")
+ self.assertEqual(result.author, "Pascal Quignard")
+ self.assertEqual(result.year, "2002")
+ self.assertEqual(result.connector, self.connector)
@responses.activate
def test_load_edition_data(self):
- ''' format url from key and make request '''
- key = 'OL1234W'
+ """ format url from key and make request """
+ key = "OL1234W"
responses.add(
responses.GET,
- 'https://openlibrary.org/works/OL1234W/editions',
- json={'hi': 'there'}
+ "https://openlibrary.org/works/OL1234W/editions",
+ json={"hi": "there"},
)
result = self.connector.load_edition_data(key)
- self.assertEqual(result, {'hi': 'there'})
-
+ self.assertEqual(result, {"hi": "there"})
@responses.activate
def test_expand_book_data(self):
- ''' given a book, get more editions '''
- work = models.Work.objects.create(
- title='Test Work', openlibrary_key='OL1234W')
- edition = models.Edition.objects.create(
- title='Test Edition', parent_work=work)
+ """ given a book, get more editions """
+ work = models.Work.objects.create(title="Test Work", openlibrary_key="OL1234W")
+ edition = models.Edition.objects.create(title="Test Edition", parent_work=work)
responses.add(
responses.GET,
- 'https://openlibrary.org/works/OL1234W/editions',
- json={'entries': []},
+ "https://openlibrary.org/works/OL1234W/editions",
+ json={"entries": []},
)
with patch(
- 'bookwyrm.connectors.abstract_connector.AbstractConnector.' \
- 'create_edition_from_data'):
+ "bookwyrm.connectors.abstract_connector.AbstractConnector."
+ "create_edition_from_data"
+ ):
self.connector.expand_book_data(edition)
self.connector.expand_book_data(work)
-
def test_get_description(self):
- ''' should do some cleanup on the description data '''
- description = get_description(self.work_data['description'])
- expected = 'First in the Old Kingdom/Abhorsen series.'
+ """ should do some cleanup on the description data """
+ description = get_description(self.work_data["description"])
+ expected = "First in the Old Kingdom/Abhorsen series."
self.assertEqual(description, expected)
-
def test_get_openlibrary_key(self):
- ''' extracts the uuid '''
- key = get_openlibrary_key('/books/OL27320736M')
- self.assertEqual(key, 'OL27320736M')
-
+ """ extracts the uuid """
+ key = get_openlibrary_key("/books/OL27320736M")
+ self.assertEqual(key, "OL27320736M")
def test_get_languages(self):
- ''' looks up languages from a list '''
- languages = get_languages(self.edition_data['languages'])
- self.assertEqual(languages, ['English'])
-
+ """ looks up languages from a list """
+ languages = get_languages(self.edition_data["languages"])
+ self.assertEqual(languages, ["English"])
def test_pick_default_edition(self):
- ''' detect if the loaded json is an edition '''
- edition = pick_default_edition(self.edition_list_data['entries'])
- self.assertEqual(edition['key'], '/books/OL9788823M')
-
+ """ detect if the loaded json is an edition """
+ edition = pick_default_edition(self.edition_list_data["entries"])
+ self.assertEqual(edition["key"], "/books/OL9788823M")
@responses.activate
def test_create_edition_from_data(self):
- ''' okay but can it actually create an edition with proper metadata '''
- work = models.Work.objects.create(title='Hello')
+ """ okay but can it actually create an edition with proper metadata """
+ work = models.Work.objects.create(title="Hello")
responses.add(
responses.GET,
- 'https://openlibrary.org/authors/OL382982A',
- json={'hi': 'there'},
- status=200)
- with patch('bookwyrm.connectors.openlibrary.Connector.' \
- 'get_authors_from_data') as mock:
+ "https://openlibrary.org/authors/OL382982A",
+ json={"hi": "there"},
+ status=200,
+ )
+ with patch(
+ "bookwyrm.connectors.openlibrary.Connector." "get_authors_from_data"
+ ) as mock:
mock.return_value = []
- result = self.connector.create_edition_from_data(
- work, self.edition_data)
+ result = self.connector.create_edition_from_data(work, self.edition_data)
self.assertEqual(result.parent_work, work)
- self.assertEqual(result.title, 'Sabriel')
- self.assertEqual(result.isbn_10, '0060273224')
+ self.assertEqual(result.title, "Sabriel")
+ self.assertEqual(result.isbn_10, "0060273224")
self.assertIsNotNone(result.description)
- self.assertEqual(result.languages[0], 'English')
- self.assertEqual(result.publishers[0], 'Harper Trophy')
+ self.assertEqual(result.languages[0], "English")
+ self.assertEqual(result.publishers[0], "Harper Trophy")
self.assertEqual(result.pages, 491)
- self.assertEqual(result.subjects[0], 'Fantasy.')
- self.assertEqual(result.physical_format, 'Hardcover')
+ self.assertEqual(result.subjects[0], "Fantasy.")
+ self.assertEqual(result.physical_format, "Hardcover")
diff --git a/bookwyrm/tests/connectors/test_self_connector.py b/bookwyrm/tests/connectors/test_self_connector.py
index 0fc78955..9925f594 100644
--- a/bookwyrm/tests/connectors/test_self_connector.py
+++ b/bookwyrm/tests/connectors/test_self_connector.py
@@ -1,4 +1,4 @@
-''' testing book data connectors '''
+""" testing book data connectors """
import datetime
from django.test import TestCase
from django.utils import timezone
@@ -9,100 +9,98 @@ from bookwyrm.settings import DOMAIN
class SelfConnector(TestCase):
- ''' just uses local data '''
+ """ just uses local data """
+
def setUp(self):
- ''' creating the connector '''
+ """ creating the connector """
models.Connector.objects.create(
identifier=DOMAIN,
- name='Local',
+ name="Local",
local=True,
- connector_file='self_connector',
- base_url='https://%s' % DOMAIN,
- books_url='https://%s/book' % DOMAIN,
- covers_url='https://%s/images/covers' % DOMAIN,
- search_url='https://%s/search?q=' % DOMAIN,
+ connector_file="self_connector",
+ base_url="https://%s" % DOMAIN,
+ books_url="https://%s/book" % DOMAIN,
+ covers_url="https://%s/images/covers" % DOMAIN,
+ search_url="https://%s/search?q=" % DOMAIN,
priority=1,
)
self.connector = Connector(DOMAIN)
-
def test_format_search_result(self):
- ''' create a SearchResult '''
- author = models.Author.objects.create(name='Anonymous')
+ """ create a SearchResult """
+ author = models.Author.objects.create(name="Anonymous")
edition = models.Edition.objects.create(
- title='Edition of Example Work',
+ title="Edition of Example Work",
published_date=datetime.datetime(1980, 5, 10, tzinfo=timezone.utc),
)
edition.authors.add(author)
- result = self.connector.search('Edition of Example')[0]
- self.assertEqual(result.title, 'Edition of Example Work')
+ result = self.connector.search("Edition of Example")[0]
+ self.assertEqual(result.title, "Edition of Example Work")
self.assertEqual(result.key, edition.remote_id)
- self.assertEqual(result.author, 'Anonymous')
+ self.assertEqual(result.author, "Anonymous")
self.assertEqual(result.year, 1980)
self.assertEqual(result.connector, self.connector)
-
def test_search_rank(self):
- ''' prioritize certain results '''
- author = models.Author.objects.create(name='Anonymous')
+ """ prioritize certain results """
+ author = models.Author.objects.create(name="Anonymous")
edition = models.Edition.objects.create(
- title='Edition of Example Work',
+ title="Edition of Example Work",
published_date=datetime.datetime(1980, 5, 10, tzinfo=timezone.utc),
- parent_work=models.Work.objects.create(title='')
+ parent_work=models.Work.objects.create(title=""),
)
# author text is rank C
edition.authors.add(author)
# series is rank D
models.Edition.objects.create(
- title='Another Edition',
- series='Anonymous',
- parent_work=models.Work.objects.create(title='')
+ title="Another Edition",
+ series="Anonymous",
+ parent_work=models.Work.objects.create(title=""),
)
# subtitle is rank B
models.Edition.objects.create(
- title='More Editions',
- subtitle='The Anonymous Edition',
- parent_work=models.Work.objects.create(title='')
+ title="More Editions",
+ subtitle="The Anonymous Edition",
+ parent_work=models.Work.objects.create(title=""),
)
# title is rank A
- models.Edition.objects.create(title='Anonymous')
+ models.Edition.objects.create(title="Anonymous")
# doesn't rank in this search
edition = models.Edition.objects.create(
- title='An Edition',
- parent_work=models.Work.objects.create(title='')
+ title="An Edition", parent_work=models.Work.objects.create(title="")
)
- results = self.connector.search('Anonymous')
+ results = self.connector.search("Anonymous")
self.assertEqual(len(results), 3)
- self.assertEqual(results[0].title, 'Anonymous')
- self.assertEqual(results[1].title, 'More Editions')
- self.assertEqual(results[2].title, 'Edition of Example Work')
-
+ self.assertEqual(results[0].title, "Anonymous")
+ self.assertEqual(results[1].title, "More Editions")
+ self.assertEqual(results[2].title, "Edition of Example Work")
def test_search_multiple_editions(self):
- ''' it should get rid of duplicate editions for the same work '''
- work = models.Work.objects.create(title='Work Title')
+ """ it should get rid of duplicate editions for the same work """
+ work = models.Work.objects.create(title="Work Title")
edition_1 = models.Edition.objects.create(
- title='Edition 1 Title', parent_work=work)
+ title="Edition 1 Title", parent_work=work
+ )
edition_2 = models.Edition.objects.create(
- title='Edition 2 Title', parent_work=work)
- edition_3 = models.Edition.objects.create(
- title='Fish', parent_work=work)
+ title="Edition 2 Title", parent_work=work
+ )
+ edition_3 = models.Edition.objects.create(title="Fish", parent_work=work)
work.default_edition = edition_2
work.save()
# pick the best edition
- results = self.connector.search('Edition 1 Title')
+ results = self.connector.search("Edition 1 Title")
self.assertEqual(len(results), 1)
self.assertEqual(results[0].key, edition_1.remote_id)
# pick the default edition when no match is best
- results = self.connector.search('Edition Title')
+ results = self.connector.search("Edition Title")
self.assertEqual(len(results), 1)
self.assertEqual(results[0].key, edition_2.remote_id)
# only matches one edition, so no deduplication takes place
- results = self.connector.search('Fish')
+ results = self.connector.search("Fish")
self.assertEqual(len(results), 1)
self.assertEqual(results[0].key, edition_3.remote_id)
diff --git a/bookwyrm/tests/data/ol_isbn_search.json b/bookwyrm/tests/data/ol_isbn_search.json
new file mode 100644
index 00000000..8516ff06
--- /dev/null
+++ b/bookwyrm/tests/data/ol_isbn_search.json
@@ -0,0 +1,45 @@
+{
+ "ISBN:9782070427796": {
+ "url": "https://openlibrary.org/books/OL16262504M/Les_ombres_errantes",
+ "key": "/books/OL16262504M",
+ "title": "Les ombres errantes",
+ "authors": [
+ {
+ "url": "https://openlibrary.org/authors/OL269675A/Pascal_Quignard",
+ "name": "Pascal Quignard"
+ }
+ ],
+ "by_statement": "Pascal Quignard.",
+ "identifiers": {
+ "goodreads": [
+ "1835483"
+ ],
+ "librarything": [
+ "983474"
+ ],
+ "isbn_10": [
+ "207042779X"
+ ],
+ "openlibrary": [
+ "OL16262504M"
+ ]
+ },
+ "classifications": {
+ "dewey_decimal_class": [
+ "848/.91403"
+ ]
+ },
+ "publishers": [
+ {
+ "name": "Gallimard"
+ }
+ ],
+ "publish_places": [
+ {
+ "name": "Paris"
+ }
+ ],
+ "publish_date": "2002",
+ "notes": "Hardback published Grasset, 2002."
+ }
+}
diff --git a/bookwyrm/tests/models/test_activitypub_mixin.py b/bookwyrm/tests/models/test_activitypub_mixin.py
index 11b944d9..930f3a53 100644
--- a/bookwyrm/tests/models/test_activitypub_mixin.py
+++ b/bookwyrm/tests/models/test_activitypub_mixin.py
@@ -1,4 +1,4 @@
-''' testing model activitypub utilities '''
+""" testing model activitypub utilities """
from unittest.mock import patch
from collections import namedtuple
from dataclasses import dataclass
@@ -12,238 +12,249 @@ from bookwyrm.models import base_model
from bookwyrm.models.activitypub_mixin import ActivitypubMixin
from bookwyrm.models.activitypub_mixin import ActivityMixin, ObjectMixin
+
class ActivitypubMixins(TestCase):
- ''' functionality shared across models '''
+ """ functionality shared across models """
+
def setUp(self):
- ''' shared data '''
+ """ shared data """
self.local_user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.com', 'mouseword',
- local=True, localname='mouse')
- self.local_user.remote_id = 'http://example.com/a/b'
+ "mouse", "mouse@mouse.com", "mouseword", local=True, localname="mouse"
+ )
+ self.local_user.remote_id = "http://example.com/a/b"
self.local_user.save(broadcast=False)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
self.object_mock = {
- 'to': 'to field', 'cc': 'cc field',
- 'content': 'hi', 'id': 'bip', 'type': 'Test',
- 'published': '2020-12-04T17:52:22.623807+00:00',
+ "to": "to field",
+ "cc": "cc field",
+ "content": "hi",
+ "id": "bip",
+ "type": "Test",
+ "published": "2020-12-04T17:52:22.623807+00:00",
}
-
# ActivitypubMixin
def test_to_activity(self):
- ''' model to ActivityPub json '''
+ """ model to ActivityPub json """
+
@dataclass(init=False)
class TestActivity(ActivityObject):
- ''' real simple mock '''
- type: str = 'Test'
+ """ real simple mock """
+
+ type: str = "Test"
class TestModel(ActivitypubMixin, base_model.BookWyrmModel):
- ''' real simple mock model because BookWyrmModel is abstract '''
+ """ real simple mock model because BookWyrmModel is abstract """
instance = TestModel()
- instance.remote_id = 'https://www.example.com/test'
+ instance.remote_id = "https://www.example.com/test"
instance.activity_serializer = TestActivity
activity = instance.to_activity()
self.assertIsInstance(activity, dict)
- self.assertEqual(activity['id'], 'https://www.example.com/test')
- self.assertEqual(activity['type'], 'Test')
-
+ self.assertEqual(activity["id"], "https://www.example.com/test")
+ self.assertEqual(activity["type"], "Test")
def test_find_existing_by_remote_id(self):
- ''' attempt to match a remote id to an object in the db '''
+ """ attempt to match a remote id to an object in the db """
# uses a different remote id scheme
# this isn't really part of this test directly but it's helpful to state
book = models.Edition.objects.create(
- title='Test Edition', remote_id='http://book.com/book')
+ title="Test Edition", remote_id="http://book.com/book"
+ )
- self.assertEqual(book.origin_id, 'http://book.com/book')
- self.assertNotEqual(book.remote_id, 'http://book.com/book')
+ self.assertEqual(book.origin_id, "http://book.com/book")
+ self.assertNotEqual(book.remote_id, "http://book.com/book")
# uses subclasses
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.Comment.objects.create(
- user=self.local_user, content='test status', book=book, \
- remote_id='https://comment.net')
+ user=self.local_user,
+ content="test status",
+ book=book,
+ remote_id="https://comment.net",
+ )
- result = models.User.find_existing_by_remote_id('hi')
+ result = models.User.find_existing_by_remote_id("hi")
self.assertIsNone(result)
- result = models.User.find_existing_by_remote_id(
- 'http://example.com/a/b')
+ result = models.User.find_existing_by_remote_id("http://example.com/a/b")
self.assertEqual(result, self.local_user)
# test using origin id
- result = models.Edition.find_existing_by_remote_id(
- 'http://book.com/book')
+ result = models.Edition.find_existing_by_remote_id("http://book.com/book")
self.assertEqual(result, book)
# test subclass match
- result = models.Status.find_existing_by_remote_id(
- 'https://comment.net')
-
+ result = models.Status.find_existing_by_remote_id("https://comment.net")
def test_find_existing(self):
- ''' match a blob of data to a model '''
+ """ match a blob of data to a model """
book = models.Edition.objects.create(
- title='Test edition',
- openlibrary_key='OL1234',
+ title="Test edition",
+ openlibrary_key="OL1234",
)
- result = models.Edition.find_existing(
- {'openlibraryKey': 'OL1234'})
+ result = models.Edition.find_existing({"openlibraryKey": "OL1234"})
self.assertEqual(result, book)
-
def test_get_recipients_public_object(self):
- ''' determines the recipients for an object's broadcast '''
- MockSelf = namedtuple('Self', ('privacy'))
- mock_self = MockSelf('public')
+ """ determines the recipients for an object's broadcast """
+ MockSelf = namedtuple("Self", ("privacy"))
+ mock_self = MockSelf("public")
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 1)
self.assertEqual(recipients[0], self.remote_user.inbox)
-
def test_get_recipients_public_user_object_no_followers(self):
- ''' determines the recipients for a user's object broadcast '''
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ """ determines the recipients for a user's object broadcast """
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 0)
-
def test_get_recipients_public_user_object(self):
- ''' determines the recipients for a user's object broadcast '''
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ """ determines the recipients for a user's object broadcast """
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
self.local_user.followers.add(self.remote_user)
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 1)
self.assertEqual(recipients[0], self.remote_user.inbox)
-
def test_get_recipients_public_user_object_with_mention(self):
- ''' determines the recipients for a user's object broadcast '''
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ """ determines the recipients for a user's object broadcast """
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
self.local_user.followers.add(self.remote_user)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
another_remote_user = models.User.objects.create_user(
- 'nutria', 'nutria@nutria.com', 'nutriaword',
+ "nutria",
+ "nutria@nutria.com",
+ "nutriaword",
local=False,
- remote_id='https://example.com/users/nutria',
- inbox='https://example.com/users/nutria/inbox',
- outbox='https://example.com/users/nutria/outbox',
+ remote_id="https://example.com/users/nutria",
+ inbox="https://example.com/users/nutria/inbox",
+ outbox="https://example.com/users/nutria/outbox",
)
- MockSelf = namedtuple('Self', ('privacy', 'user', 'recipients'))
- mock_self = MockSelf('public', self.local_user, [another_remote_user])
+ MockSelf = namedtuple("Self", ("privacy", "user", "recipients"))
+ mock_self = MockSelf("public", self.local_user, [another_remote_user])
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 2)
self.assertEqual(recipients[0], another_remote_user.inbox)
self.assertEqual(recipients[1], self.remote_user.inbox)
-
def test_get_recipients_direct(self):
- ''' determines the recipients for a user's object broadcast '''
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ """ determines the recipients for a user's object broadcast """
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
self.local_user.followers.add(self.remote_user)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
another_remote_user = models.User.objects.create_user(
- 'nutria', 'nutria@nutria.com', 'nutriaword',
+ "nutria",
+ "nutria@nutria.com",
+ "nutriaword",
local=False,
- remote_id='https://example.com/users/nutria',
- inbox='https://example.com/users/nutria/inbox',
- outbox='https://example.com/users/nutria/outbox',
+ remote_id="https://example.com/users/nutria",
+ inbox="https://example.com/users/nutria/inbox",
+ outbox="https://example.com/users/nutria/outbox",
)
- MockSelf = namedtuple('Self', ('privacy', 'user', 'recipients'))
- mock_self = MockSelf('direct', self.local_user, [another_remote_user])
+ MockSelf = namedtuple("Self", ("privacy", "user", "recipients"))
+ mock_self = MockSelf("direct", self.local_user, [another_remote_user])
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 1)
self.assertEqual(recipients[0], another_remote_user.inbox)
-
def test_get_recipients_combine_inboxes(self):
- ''' should combine users with the same shared_inbox '''
- self.remote_user.shared_inbox = 'http://example.com/inbox'
+ """ should combine users with the same shared_inbox """
+ self.remote_user.shared_inbox = "http://example.com/inbox"
self.remote_user.save(broadcast=False)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
another_remote_user = models.User.objects.create_user(
- 'nutria', 'nutria@nutria.com', 'nutriaword',
+ "nutria",
+ "nutria@nutria.com",
+ "nutriaword",
local=False,
- remote_id='https://example.com/users/nutria',
- inbox='https://example.com/users/nutria/inbox',
- shared_inbox='http://example.com/inbox',
- outbox='https://example.com/users/nutria/outbox',
+ remote_id="https://example.com/users/nutria",
+ inbox="https://example.com/users/nutria/inbox",
+ shared_inbox="http://example.com/inbox",
+ outbox="https://example.com/users/nutria/outbox",
)
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
self.local_user.followers.add(self.remote_user)
self.local_user.followers.add(another_remote_user)
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 1)
- self.assertEqual(recipients[0], 'http://example.com/inbox')
-
+ self.assertEqual(recipients[0], "http://example.com/inbox")
def test_get_recipients_software(self):
- ''' should differentiate between bookwyrm and other remote users '''
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ """ should differentiate between bookwyrm and other remote users """
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
another_remote_user = models.User.objects.create_user(
- 'nutria', 'nutria@nutria.com', 'nutriaword',
+ "nutria",
+ "nutria@nutria.com",
+ "nutriaword",
local=False,
- remote_id='https://example.com/users/nutria',
- inbox='https://example.com/users/nutria/inbox',
- outbox='https://example.com/users/nutria/outbox',
+ remote_id="https://example.com/users/nutria",
+ inbox="https://example.com/users/nutria/inbox",
+ outbox="https://example.com/users/nutria/outbox",
bookwyrm_user=False,
)
- MockSelf = namedtuple('Self', ('privacy', 'user'))
- mock_self = MockSelf('public', self.local_user)
+ MockSelf = namedtuple("Self", ("privacy", "user"))
+ mock_self = MockSelf("public", self.local_user)
self.local_user.followers.add(self.remote_user)
self.local_user.followers.add(another_remote_user)
recipients = ActivitypubMixin.get_recipients(mock_self)
self.assertEqual(len(recipients), 2)
- recipients = ActivitypubMixin.get_recipients(
- mock_self, software='bookwyrm')
+ recipients = ActivitypubMixin.get_recipients(mock_self, software="bookwyrm")
self.assertEqual(len(recipients), 1)
self.assertEqual(recipients[0], self.remote_user.inbox)
- recipients = ActivitypubMixin.get_recipients(
- mock_self, software='other')
+ recipients = ActivitypubMixin.get_recipients(mock_self, software="other")
self.assertEqual(len(recipients), 1)
self.assertEqual(recipients[0], another_remote_user.inbox)
-
# ObjectMixin
def test_object_save_create(self):
- ''' should save uneventufully when broadcast is disabled '''
+ """ should save uneventufully when broadcast is disabled """
+
class Success(Exception):
- ''' this means we got to the right method '''
+ """ this means we got to the right method """
class ObjectModel(ObjectMixin, base_model.BookWyrmModel):
- ''' real simple mock model because BookWyrmModel is abstract '''
- user = models.fields.ForeignKey('User', on_delete=db.models.CASCADE)
+ """ real simple mock model because BookWyrmModel is abstract """
+
+ user = models.fields.ForeignKey("User", on_delete=db.models.CASCADE)
+
def save(self, *args, **kwargs):
- with patch('django.db.models.Model.save'):
+ with patch("django.db.models.Model.save"):
super().save(*args, **kwargs)
- def broadcast(self, activity, sender, **kwargs):#pylint: disable=arguments-differ
- ''' do something '''
+
+ def broadcast(
+ self, activity, sender, **kwargs
+ ): # pylint: disable=arguments-differ
+ """ do something """
raise Success()
- def to_create_activity(self, user):#pylint: disable=arguments-differ
+
+ def to_create_activity(self, user): # pylint: disable=arguments-differ
return {}
with self.assertRaises(Success):
@@ -253,20 +264,24 @@ class ActivitypubMixins(TestCase):
ObjectModel(user=self.local_user).save(broadcast=False)
ObjectModel(user=None).save()
-
def test_object_save_update(self):
- ''' should save uneventufully when broadcast is disabled '''
+ """ should save uneventufully when broadcast is disabled """
+
class Success(Exception):
- ''' this means we got to the right method '''
+ """ this means we got to the right method """
class UpdateObjectModel(ObjectMixin, base_model.BookWyrmModel):
- ''' real simple mock model because BookWyrmModel is abstract '''
- user = models.fields.ForeignKey('User', on_delete=db.models.CASCADE)
+ """ real simple mock model because BookWyrmModel is abstract """
+
+ user = models.fields.ForeignKey("User", on_delete=db.models.CASCADE)
last_edited_by = models.fields.ForeignKey(
- 'User', on_delete=db.models.CASCADE)
+ "User", on_delete=db.models.CASCADE
+ )
+
def save(self, *args, **kwargs):
- with patch('django.db.models.Model.save'):
+ with patch("django.db.models.Model.save"):
super().save(*args, **kwargs)
+
def to_update_activity(self, user):
raise Success()
@@ -275,87 +290,71 @@ class ActivitypubMixins(TestCase):
with self.assertRaises(Success):
UpdateObjectModel(id=1, last_edited_by=self.local_user).save()
-
def test_object_save_delete(self):
- ''' should create delete activities when objects are deleted by flag '''
+ """ should create delete activities when objects are deleted by flag """
+
class ActivitySuccess(Exception):
- ''' this means we got to the right method '''
+ """ this means we got to the right method """
class DeletableObjectModel(ObjectMixin, base_model.BookWyrmModel):
- ''' real simple mock model because BookWyrmModel is abstract '''
- user = models.fields.ForeignKey('User', on_delete=db.models.CASCADE)
+ """ real simple mock model because BookWyrmModel is abstract """
+
+ user = models.fields.ForeignKey("User", on_delete=db.models.CASCADE)
deleted = models.fields.BooleanField()
+
def save(self, *args, **kwargs):
- with patch('django.db.models.Model.save'):
+ with patch("django.db.models.Model.save"):
super().save(*args, **kwargs)
+
def to_delete_activity(self, user):
raise ActivitySuccess()
with self.assertRaises(ActivitySuccess):
- DeletableObjectModel(
- id=1, user=self.local_user, deleted=True).save()
-
+ DeletableObjectModel(id=1, user=self.local_user, deleted=True).save()
def test_to_delete_activity(self):
- ''' wrapper for Delete activity '''
- MockSelf = namedtuple('Self', ('remote_id', 'to_activity'))
+ """ wrapper for Delete activity """
+ MockSelf = namedtuple("Self", ("remote_id", "to_activity"))
mock_self = MockSelf(
- 'https://example.com/status/1',
- lambda *args: self.object_mock
+ "https://example.com/status/1", lambda *args: self.object_mock
)
- activity = ObjectMixin.to_delete_activity(
- mock_self, self.local_user)
+ activity = ObjectMixin.to_delete_activity(mock_self, self.local_user)
+ self.assertEqual(activity["id"], "https://example.com/status/1/activity")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["type"], "Delete")
+ self.assertEqual(activity["to"], ["%s/followers" % self.local_user.remote_id])
self.assertEqual(
- activity['id'],
- 'https://example.com/status/1/activity'
+ activity["cc"], ["https://www.w3.org/ns/activitystreams#Public"]
)
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Delete')
- self.assertEqual(
- activity['to'],
- ['%s/followers' % self.local_user.remote_id])
- self.assertEqual(
- activity['cc'],
- ['https://www.w3.org/ns/activitystreams#Public'])
-
def test_to_update_activity(self):
- ''' ditto above but for Update '''
- MockSelf = namedtuple('Self', ('remote_id', 'to_activity'))
+ """ ditto above but for Update """
+ MockSelf = namedtuple("Self", ("remote_id", "to_activity"))
mock_self = MockSelf(
- 'https://example.com/status/1',
- lambda *args: self.object_mock
+ "https://example.com/status/1", lambda *args: self.object_mock
)
- activity = ObjectMixin.to_update_activity(
- mock_self, self.local_user)
+ activity = ObjectMixin.to_update_activity(mock_self, self.local_user)
self.assertIsNotNone(
- re.match(
- r'^https:\/\/example\.com\/status\/1#update\/.*',
- activity['id']
- )
+ re.match(r"^https:\/\/example\.com\/status\/1#update\/.*", activity["id"])
)
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Update')
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["type"], "Update")
self.assertEqual(
- activity['to'],
- ['https://www.w3.org/ns/activitystreams#Public'])
- self.assertIsInstance(activity['object'], dict)
-
+ activity["to"], ["https://www.w3.org/ns/activitystreams#Public"]
+ )
+ self.assertIsInstance(activity["object"], dict)
# Activity mixin
def test_to_undo_activity(self):
- ''' and again, for Undo '''
- MockSelf = namedtuple('Self', ('remote_id', 'to_activity', 'user'))
+ """ and again, for Undo """
+ MockSelf = namedtuple("Self", ("remote_id", "to_activity", "user"))
mock_self = MockSelf(
- 'https://example.com/status/1',
+ "https://example.com/status/1",
lambda *args: self.object_mock,
self.local_user,
)
activity = ActivityMixin.to_undo_activity(mock_self)
- self.assertEqual(
- activity['id'],
- 'https://example.com/status/1#undo'
- )
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Undo')
- self.assertIsInstance(activity['object'], dict)
+ self.assertEqual(activity["id"], "https://example.com/status/1#undo")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["type"], "Undo")
+ self.assertIsInstance(activity["object"], dict)
diff --git a/bookwyrm/tests/models/test_base_model.py b/bookwyrm/tests/models/test_base_model.py
index ab388efe..4479d156 100644
--- a/bookwyrm/tests/models/test_base_model.py
+++ b/bookwyrm/tests/models/test_base_model.py
@@ -1,42 +1,41 @@
-''' testing models '''
+""" testing models """
from django.test import TestCase
from bookwyrm import models
from bookwyrm.models import base_model
from bookwyrm.settings import DOMAIN
+
class BaseModel(TestCase):
- ''' functionality shared across models '''
+ """ functionality shared across models """
+
def test_remote_id(self):
- ''' these should be generated '''
+ """ these should be generated """
instance = base_model.BookWyrmModel()
instance.id = 1
expected = instance.get_remote_id()
- self.assertEqual(expected, 'https://%s/bookwyrmmodel/1' % DOMAIN)
+ self.assertEqual(expected, "https://%s/bookwyrmmodel/1" % DOMAIN)
def test_remote_id_with_user(self):
- ''' format of remote id when there's a user object '''
+ """ format of remote id when there's a user object """
user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.com', 'mouseword',
- local=True, localname='mouse')
+ "mouse", "mouse@mouse.com", "mouseword", local=True, localname="mouse"
+ )
instance = base_model.BookWyrmModel()
instance.user = user
instance.id = 1
expected = instance.get_remote_id()
- self.assertEqual(
- expected,
- 'https://%s/user/mouse/bookwyrmmodel/1' % DOMAIN)
+ self.assertEqual(expected, "https://%s/user/mouse/bookwyrmmodel/1" % DOMAIN)
def test_execute_after_save(self):
- ''' this function sets remote ids after creation '''
+ """ this function sets remote ids after creation """
# using Work because it BookWrymModel is abstract and this requires save
# Work is a relatively not-fancy model.
- instance = models.Work.objects.create(title='work title')
+ instance = models.Work.objects.create(title="work title")
instance.remote_id = None
base_model.execute_after_save(None, instance, True)
self.assertEqual(
- instance.remote_id,
- 'https://%s/book/%d' % (DOMAIN, instance.id)
+ instance.remote_id, "https://%s/book/%d" % (DOMAIN, instance.id)
)
# shouldn't set remote_id if it's not created
diff --git a/bookwyrm/tests/models/test_book_model.py b/bookwyrm/tests/models/test_book_model.py
index 98d6d446..14ab0c57 100644
--- a/bookwyrm/tests/models/test_book_model.py
+++ b/bookwyrm/tests/models/test_book_model.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from dateutil.parser import parse
from django.test import TestCase
from django.utils import timezone
@@ -8,88 +8,80 @@ from bookwyrm.models.book import isbn_10_to_13, isbn_13_to_10
class Book(TestCase):
- ''' not too much going on in the books model but here we are '''
+ """ not too much going on in the books model but here we are """
+
def setUp(self):
- ''' we'll need some books '''
+ """ we'll need some books """
self.work = models.Work.objects.create(
- title='Example Work',
- remote_id='https://example.com/book/1'
+ title="Example Work", remote_id="https://example.com/book/1"
)
self.first_edition = models.Edition.objects.create(
- title='Example Edition',
+ title="Example Edition",
parent_work=self.work,
)
self.second_edition = models.Edition.objects.create(
- title='Another Example Edition',
+ title="Another Example Edition",
parent_work=self.work,
)
def test_remote_id(self):
- ''' fanciness with remote/origin ids '''
- remote_id = 'https://%s/book/%d' % (settings.DOMAIN, self.work.id)
+ """ fanciness with remote/origin ids """
+ remote_id = "https://%s/book/%d" % (settings.DOMAIN, self.work.id)
self.assertEqual(self.work.get_remote_id(), remote_id)
self.assertEqual(self.work.remote_id, remote_id)
def test_create_book(self):
- ''' you shouldn't be able to create Books (only editions and works) '''
- self.assertRaises(
- ValueError,
- models.Book.objects.create,
- title='Invalid Book'
- )
+ """ you shouldn't be able to create Books (only editions and works) """
+ self.assertRaises(ValueError, models.Book.objects.create, title="Invalid Book")
def test_isbn_10_to_13(self):
- ''' checksums and so on '''
- isbn_10 = '178816167X'
+ """ checksums and so on """
+ isbn_10 = "178816167X"
isbn_13 = isbn_10_to_13(isbn_10)
- self.assertEqual(isbn_13, '9781788161671')
+ self.assertEqual(isbn_13, "9781788161671")
- isbn_10 = '1-788-16167-X'
+ isbn_10 = "1-788-16167-X"
isbn_13 = isbn_10_to_13(isbn_10)
- self.assertEqual(isbn_13, '9781788161671')
-
+ self.assertEqual(isbn_13, "9781788161671")
def test_isbn_13_to_10(self):
- ''' checksums and so on '''
- isbn_13 = '9781788161671'
+ """ checksums and so on """
+ isbn_13 = "9781788161671"
isbn_10 = isbn_13_to_10(isbn_13)
- self.assertEqual(isbn_10, '178816167X')
+ self.assertEqual(isbn_10, "178816167X")
- isbn_13 = '978-1788-16167-1'
+ isbn_13 = "978-1788-16167-1"
isbn_10 = isbn_13_to_10(isbn_13)
- self.assertEqual(isbn_10, '178816167X')
-
+ self.assertEqual(isbn_10, "178816167X")
def test_get_edition_info(self):
- ''' text slug about an edition '''
- book = models.Edition.objects.create(title='Test Edition')
- self.assertEqual(book.edition_info, '')
+ """ text slug about an edition """
+ book = models.Edition.objects.create(title="Test Edition")
+ self.assertEqual(book.edition_info, "")
- book.physical_format = 'worm'
+ book.physical_format = "worm"
book.save()
- self.assertEqual(book.edition_info, 'worm')
+ self.assertEqual(book.edition_info, "worm")
- book.languages = ['English']
+ book.languages = ["English"]
book.save()
- self.assertEqual(book.edition_info, 'worm')
+ self.assertEqual(book.edition_info, "worm")
- book.languages = ['Glorbish', 'English']
+ book.languages = ["Glorbish", "English"]
book.save()
- self.assertEqual(book.edition_info, 'worm, Glorbish language')
+ self.assertEqual(book.edition_info, "worm, Glorbish language")
- book.published_date = timezone.make_aware(parse('2020'))
+ book.published_date = timezone.make_aware(parse("2020"))
book.save()
- self.assertEqual(book.edition_info, 'worm, Glorbish language, 2020')
- self.assertEqual(
- book.alt_text, 'Test Edition cover (worm, Glorbish language, 2020)')
-
+ self.assertEqual(book.edition_info, "worm, Glorbish language, 2020")
+ self.assertEqual(book.alt_text, "Test Edition (worm, Glorbish language, 2020)")
def test_get_rank(self):
- ''' sets the data quality index for the book '''
+ """ sets the data quality index for the book """
# basic rank
self.assertEqual(self.first_edition.edition_rank, 0)
- self.first_edition.description = 'hi'
+ self.first_edition.description = "hi"
self.first_edition.save()
self.assertEqual(self.first_edition.edition_rank, 1)
diff --git a/bookwyrm/tests/models/test_fields.py b/bookwyrm/tests/models/test_fields.py
index 24c0fb02..522d16f9 100644
--- a/bookwyrm/tests/models/test_fields.py
+++ b/bookwyrm/tests/models/test_fields.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from io import BytesIO
from collections import namedtuple
from dataclasses import dataclass
@@ -23,408 +23,388 @@ from bookwyrm.models import fields, User, Status
from bookwyrm.models.base_model import BookWyrmModel
from bookwyrm.models.activitypub_mixin import ActivitypubMixin
-#pylint: disable=too-many-public-methods
+# pylint: disable=too-many-public-methods
class ActivitypubFields(TestCase):
- ''' overwrites standard model feilds to work with activitypub '''
+ """ overwrites standard model feilds to work with activitypub """
+
def test_validate_remote_id(self):
- ''' should look like a url '''
- self.assertIsNone(fields.validate_remote_id('http://www.example.com'))
- self.assertIsNone(fields.validate_remote_id('https://www.example.com'))
- self.assertIsNone(fields.validate_remote_id('http://exle.com/dlg-23/x'))
+ """ should look like a url """
+ self.assertIsNone(fields.validate_remote_id("http://www.example.com"))
+ self.assertIsNone(fields.validate_remote_id("https://www.example.com"))
+ self.assertIsNone(fields.validate_remote_id("http://exle.com/dlg-23/x"))
self.assertRaises(
- ValidationError, fields.validate_remote_id,
- 'http:/example.com/dlfjg-23/x')
+ ValidationError, fields.validate_remote_id, "http:/example.com/dlfjg-23/x"
+ )
self.assertRaises(
- ValidationError, fields.validate_remote_id,
- 'www.example.com/dlfjg-23/x')
+ ValidationError, fields.validate_remote_id, "www.example.com/dlfjg-23/x"
+ )
self.assertRaises(
- ValidationError, fields.validate_remote_id,
- 'http://www.example.com/dlfjg 23/x')
+ ValidationError,
+ fields.validate_remote_id,
+ "http://www.example.com/dlfjg 23/x",
+ )
def test_activitypub_field_mixin(self):
- ''' generic mixin with super basic to and from functionality '''
+ """ generic mixin with super basic to and from functionality """
instance = fields.ActivitypubFieldMixin()
- self.assertEqual(instance.field_to_activity('fish'), 'fish')
- self.assertEqual(instance.field_from_activity('fish'), 'fish')
+ self.assertEqual(instance.field_to_activity("fish"), "fish")
+ self.assertEqual(instance.field_from_activity("fish"), "fish")
self.assertFalse(instance.deduplication_field)
instance = fields.ActivitypubFieldMixin(
- activitypub_wrapper='endpoints', activitypub_field='outbox'
+ activitypub_wrapper="endpoints", activitypub_field="outbox"
)
- self.assertEqual(
- instance.field_to_activity('fish'),
- {'outbox': 'fish'}
- )
- self.assertEqual(
- instance.field_from_activity({'outbox': 'fish'}),
- 'fish'
- )
- self.assertEqual(instance.get_activitypub_field(), 'endpoints')
+ self.assertEqual(instance.field_to_activity("fish"), {"outbox": "fish"})
+ self.assertEqual(instance.field_from_activity({"outbox": "fish"}), "fish")
+ self.assertEqual(instance.get_activitypub_field(), "endpoints")
instance = fields.ActivitypubFieldMixin()
- instance.name = 'snake_case_name'
- self.assertEqual(instance.get_activitypub_field(), 'snakeCaseName')
+ instance.name = "snake_case_name"
+ self.assertEqual(instance.get_activitypub_field(), "snakeCaseName")
def test_set_field_from_activity(self):
- ''' setter from entire json blob '''
+ """ setter from entire json blob """
+
@dataclass
class TestModel:
- ''' real simple mock '''
+ """ real simple mock """
+
field_name: str
- mock_model = TestModel(field_name='bip')
- TestActivity = namedtuple('test', ('fieldName', 'unrelated'))
- data = TestActivity(fieldName='hi', unrelated='bfkjh')
+ mock_model = TestModel(field_name="bip")
+ TestActivity = namedtuple("test", ("fieldName", "unrelated"))
+ data = TestActivity(fieldName="hi", unrelated="bfkjh")
instance = fields.ActivitypubFieldMixin()
- instance.name = 'field_name'
+ instance.name = "field_name"
instance.set_field_from_activity(mock_model, data)
- self.assertEqual(mock_model.field_name, 'hi')
+ self.assertEqual(mock_model.field_name, "hi")
def test_set_activity_from_field(self):
- ''' set json field given entire model '''
+ """ set json field given entire model """
+
@dataclass
class TestModel:
- ''' real simple mock '''
+ """ real simple mock """
+
field_name: str
unrelated: str
- mock_model = TestModel(field_name='bip', unrelated='field')
+
+ mock_model = TestModel(field_name="bip", unrelated="field")
instance = fields.ActivitypubFieldMixin()
- instance.name = 'field_name'
+ instance.name = "field_name"
data = {}
instance.set_activity_from_field(data, mock_model)
- self.assertEqual(data['fieldName'], 'bip')
+ self.assertEqual(data["fieldName"], "bip")
def test_remote_id_field(self):
- ''' just sets some defaults on charfield '''
+ """ just sets some defaults on charfield """
instance = fields.RemoteIdField()
self.assertEqual(instance.max_length, 255)
self.assertTrue(instance.deduplication_field)
with self.assertRaises(ValidationError):
- instance.run_validators('http://www.example.com/dlfjg 23/x')
+ instance.run_validators("http://www.example.com/dlfjg 23/x")
def test_username_field(self):
- ''' again, just setting defaults on username field '''
+ """ again, just setting defaults on username field """
instance = fields.UsernameField()
- self.assertEqual(instance.activitypub_field, 'preferredUsername')
+ self.assertEqual(instance.activitypub_field, "preferredUsername")
self.assertEqual(instance.max_length, 150)
self.assertEqual(instance.unique, True)
with self.assertRaises(ValidationError):
- instance.run_validators('mouse')
- instance.run_validators('mouseexample.com')
- instance.run_validators('mouse@example.c')
- instance.run_validators('@example.com')
- instance.run_validators('mouse@examplecom')
- instance.run_validators('one two@fish.aaaa')
- instance.run_validators('a*&@exampke.com')
- instance.run_validators('trailingwhite@example.com ')
- self.assertIsNone(instance.run_validators('mouse@example.com'))
- self.assertIsNone(instance.run_validators('mo-2use@ex3ample.com'))
- self.assertIsNone(instance.run_validators('aksdhf@sdkjf-df.cm'))
-
- self.assertEqual(instance.field_to_activity('test@example.com'), 'test')
+ instance.run_validators("mouse")
+ instance.run_validators("mouseexample.com")
+ instance.run_validators("mouse@example.c")
+ instance.run_validators("@example.com")
+ instance.run_validators("mouse@examplecom")
+ instance.run_validators("one two@fish.aaaa")
+ instance.run_validators("a*&@exampke.com")
+ instance.run_validators("trailingwhite@example.com ")
+ self.assertIsNone(instance.run_validators("mouse@example.com"))
+ self.assertIsNone(instance.run_validators("mo-2use@ex3ample.com"))
+ self.assertIsNone(instance.run_validators("aksdhf@sdkjf-df.cm"))
+ self.assertEqual(instance.field_to_activity("test@example.com"), "test")
def test_privacy_field_defaults(self):
- ''' post privacy field's many default values '''
+ """ post privacy field's many default values """
instance = fields.PrivacyField()
self.assertEqual(instance.max_length, 255)
self.assertEqual(
[c[0] for c in instance.choices],
- ['public', 'unlisted', 'followers', 'direct'])
- self.assertEqual(instance.default, 'public')
+ ["public", "unlisted", "followers", "direct"],
+ )
+ self.assertEqual(instance.default, "public")
self.assertEqual(
- instance.public, 'https://www.w3.org/ns/activitystreams#Public')
+ instance.public, "https://www.w3.org/ns/activitystreams#Public"
+ )
def test_privacy_field_set_field_from_activity(self):
- ''' translate between to/cc fields and privacy '''
+ """ translate between to/cc fields and privacy """
+
@dataclass(init=False)
class TestActivity(ActivityObject):
- ''' real simple mock '''
+ """ real simple mock """
+
to: List[str]
cc: List[str]
- id: str = 'http://hi.com'
- type: str = 'Test'
+ id: str = "http://hi.com"
+ type: str = "Test"
class TestPrivacyModel(ActivitypubMixin, BookWyrmModel):
- ''' real simple mock model because BookWyrmModel is abstract '''
+ """ real simple mock model because BookWyrmModel is abstract """
+
privacy_field = fields.PrivacyField()
mention_users = fields.TagField(User)
user = fields.ForeignKey(User, on_delete=models.CASCADE)
- public = 'https://www.w3.org/ns/activitystreams#Public'
+ public = "https://www.w3.org/ns/activitystreams#Public"
data = TestActivity(
to=[public],
- cc=['bleh'],
+ cc=["bleh"],
)
- model_instance = TestPrivacyModel(privacy_field='direct')
- self.assertEqual(model_instance.privacy_field, 'direct')
+ model_instance = TestPrivacyModel(privacy_field="direct")
+ self.assertEqual(model_instance.privacy_field, "direct")
instance = fields.PrivacyField()
- instance.name = 'privacy_field'
+ instance.name = "privacy_field"
instance.set_field_from_activity(model_instance, data)
- self.assertEqual(model_instance.privacy_field, 'public')
+ self.assertEqual(model_instance.privacy_field, "public")
- data.to = ['bleh']
+ data.to = ["bleh"]
data.cc = []
instance.set_field_from_activity(model_instance, data)
- self.assertEqual(model_instance.privacy_field, 'direct')
+ self.assertEqual(model_instance.privacy_field, "direct")
- data.to = ['bleh']
- data.cc = [public, 'waah']
+ data.to = ["bleh"]
+ data.cc = [public, "waah"]
instance.set_field_from_activity(model_instance, data)
- self.assertEqual(model_instance.privacy_field, 'unlisted')
+ self.assertEqual(model_instance.privacy_field, "unlisted")
-
- @patch('bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast')
+ @patch("bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast")
def test_privacy_field_set_activity_from_field(self, _):
- ''' translate between to/cc fields and privacy '''
+ """ translate between to/cc fields and privacy """
user = User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- local=True, localname='rat')
- public = 'https://www.w3.org/ns/activitystreams#Public'
- followers = '%s/followers' % user.remote_id
+ "rat", "rat@rat.rat", "ratword", local=True, localname="rat"
+ )
+ public = "https://www.w3.org/ns/activitystreams#Public"
+ followers = "%s/followers" % user.remote_id
instance = fields.PrivacyField()
- instance.name = 'privacy_field'
+ instance.name = "privacy_field"
- model_instance = Status.objects.create(user=user, content='hi')
+ model_instance = Status.objects.create(user=user, content="hi")
activity = {}
instance.set_activity_from_field(activity, model_instance)
- self.assertEqual(activity['to'], [public])
- self.assertEqual(activity['cc'], [followers])
+ self.assertEqual(activity["to"], [public])
+ self.assertEqual(activity["cc"], [followers])
model_instance = Status.objects.create(
- user=user, content='hi', privacy='unlisted')
+ user=user, content="hi", privacy="unlisted"
+ )
activity = {}
instance.set_activity_from_field(activity, model_instance)
- self.assertEqual(activity['to'], [followers])
- self.assertEqual(activity['cc'], [public])
+ self.assertEqual(activity["to"], [followers])
+ self.assertEqual(activity["cc"], [public])
model_instance = Status.objects.create(
- user=user, content='hi', privacy='followers')
+ user=user, content="hi", privacy="followers"
+ )
activity = {}
instance.set_activity_from_field(activity, model_instance)
- self.assertEqual(activity['to'], [followers])
- self.assertEqual(activity['cc'], [])
+ self.assertEqual(activity["to"], [followers])
+ self.assertEqual(activity["cc"], [])
model_instance = Status.objects.create(
user=user,
- content='hi',
- privacy='direct',
+ content="hi",
+ privacy="direct",
)
model_instance.mention_users.set([user])
activity = {}
instance.set_activity_from_field(activity, model_instance)
- self.assertEqual(activity['to'], [user.remote_id])
- self.assertEqual(activity['cc'], [])
-
+ self.assertEqual(activity["to"], [user.remote_id])
+ self.assertEqual(activity["cc"], [])
def test_foreign_key(self):
- ''' should be able to format a related model '''
- instance = fields.ForeignKey('User', on_delete=models.CASCADE)
- Serializable = namedtuple('Serializable', ('to_activity', 'remote_id'))
- item = Serializable(lambda: {'a': 'b'}, 'https://e.b/c')
+ """ should be able to format a related model """
+ instance = fields.ForeignKey("User", on_delete=models.CASCADE)
+ Serializable = namedtuple("Serializable", ("to_activity", "remote_id"))
+ item = Serializable(lambda: {"a": "b"}, "https://e.b/c")
# returns the remote_id field of the related object
- self.assertEqual(instance.field_to_activity(item), 'https://e.b/c')
-
+ self.assertEqual(instance.field_to_activity(item), "https://e.b/c")
@responses.activate
def test_foreign_key_from_activity_str(self):
- ''' create a new object from a foreign key '''
+ """ create a new object from a foreign key """
instance = fields.ForeignKey(User, on_delete=models.CASCADE)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
userdata = json.loads(datafile.read_bytes())
# don't try to load the user icon
- del userdata['icon']
+ del userdata["icon"]
# it shouldn't match with this unrelated user:
unrelated_user = User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- local=True, localname='rat')
+ "rat", "rat@rat.rat", "ratword", local=True, localname="rat"
+ )
# test receiving an unknown remote id and loading data
responses.add(
- responses.GET,
- 'https://example.com/user/mouse',
- json=userdata,
- status=200)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
- value = instance.field_from_activity(
- 'https://example.com/user/mouse')
+ responses.GET, "https://example.com/user/mouse", json=userdata, status=200
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
+ value = instance.field_from_activity("https://example.com/user/mouse")
self.assertIsInstance(value, User)
self.assertNotEqual(value, unrelated_user)
- self.assertEqual(value.remote_id, 'https://example.com/user/mouse')
- self.assertEqual(value.name, 'MOUSE?? MOUSE!!')
-
+ self.assertEqual(value.remote_id, "https://example.com/user/mouse")
+ self.assertEqual(value.name, "MOUSE?? MOUSE!!")
def test_foreign_key_from_activity_dict(self):
- ''' test recieving activity json '''
+ """ test recieving activity json """
instance = fields.ForeignKey(User, on_delete=models.CASCADE)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
userdata = json.loads(datafile.read_bytes())
# don't try to load the user icon
- del userdata['icon']
+ del userdata["icon"]
# it shouldn't match with this unrelated user:
unrelated_user = User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- local=True, localname='rat')
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ "rat", "rat@rat.rat", "ratword", local=True, localname="rat"
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
value = instance.field_from_activity(activitypub.Person(**userdata))
self.assertIsInstance(value, User)
self.assertNotEqual(value, unrelated_user)
- self.assertEqual(value.remote_id, 'https://example.com/user/mouse')
- self.assertEqual(value.name, 'MOUSE?? MOUSE!!')
+ self.assertEqual(value.remote_id, "https://example.com/user/mouse")
+ self.assertEqual(value.name, "MOUSE?? MOUSE!!")
# et cetera but we're not testing serializing user json
-
def test_foreign_key_from_activity_dict_existing(self):
- ''' test receiving a dict of an existing object in the db '''
+ """ test receiving a dict of an existing object in the db """
instance = fields.ForeignKey(User, on_delete=models.CASCADE)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json'
- )
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
userdata = json.loads(datafile.read_bytes())
user = User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- user.remote_id = 'https://example.com/user/mouse'
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
+ user.remote_id = "https://example.com/user/mouse"
user.save(broadcast=False)
User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- local=True, localname='rat')
+ "rat", "rat@rat.rat", "ratword", local=True, localname="rat"
+ )
- with patch('bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast'):
+ with patch("bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast"):
value = instance.field_from_activity(activitypub.Person(**userdata))
self.assertEqual(value, user)
-
def test_foreign_key_from_activity_str_existing(self):
- ''' test receiving a remote id of an existing object in the db '''
+ """ test receiving a remote id of an existing object in the db """
instance = fields.ForeignKey(User, on_delete=models.CASCADE)
user = User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- local=True, localname='rat')
+ "rat", "rat@rat.rat", "ratword", local=True, localname="rat"
+ )
value = instance.field_from_activity(user.remote_id)
self.assertEqual(value, user)
-
def test_one_to_one_field(self):
- ''' a gussied up foreign key '''
- instance = fields.OneToOneField('User', on_delete=models.CASCADE)
- Serializable = namedtuple('Serializable', ('to_activity', 'remote_id'))
- item = Serializable(lambda: {'a': 'b'}, 'https://e.b/c')
- self.assertEqual(instance.field_to_activity(item), {'a': 'b'})
+ """ a gussied up foreign key """
+ instance = fields.OneToOneField("User", on_delete=models.CASCADE)
+ Serializable = namedtuple("Serializable", ("to_activity", "remote_id"))
+ item = Serializable(lambda: {"a": "b"}, "https://e.b/c")
+ self.assertEqual(instance.field_to_activity(item), {"a": "b"})
def test_many_to_many_field(self):
- ''' lists! '''
- instance = fields.ManyToManyField('User')
+ """ lists! """
+ instance = fields.ManyToManyField("User")
- Serializable = namedtuple('Serializable', ('to_activity', 'remote_id'))
- Queryset = namedtuple('Queryset', ('all', 'instance'))
- item = Serializable(lambda: {'a': 'b'}, 'https://e.b/c')
- another_item = Serializable(lambda: {}, 'example.com')
+ Serializable = namedtuple("Serializable", ("to_activity", "remote_id"))
+ Queryset = namedtuple("Queryset", ("all", "instance"))
+ item = Serializable(lambda: {"a": "b"}, "https://e.b/c")
+ another_item = Serializable(lambda: {}, "example.com")
items = Queryset(lambda: [item], another_item)
- self.assertEqual(instance.field_to_activity(items), ['https://e.b/c'])
+ self.assertEqual(instance.field_to_activity(items), ["https://e.b/c"])
- instance = fields.ManyToManyField('User', link_only=True)
- instance.name = 'snake_case'
- self.assertEqual(
- instance.field_to_activity(items),
- 'example.com/snake_case'
- )
+ instance = fields.ManyToManyField("User", link_only=True)
+ instance.name = "snake_case"
+ self.assertEqual(instance.field_to_activity(items), "example.com/snake_case")
@responses.activate
def test_many_to_many_field_from_activity(self):
- ''' resolve related fields for a list, takes a list of remote ids '''
+ """ resolve related fields for a list, takes a list of remote ids """
instance = fields.ManyToManyField(User)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json'
- )
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
userdata = json.loads(datafile.read_bytes())
# don't try to load the user icon
- del userdata['icon']
+ del userdata["icon"]
# test receiving an unknown remote id and loading data
responses.add(
- responses.GET,
- 'https://example.com/user/mouse',
- json=userdata,
- status=200)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ responses.GET, "https://example.com/user/mouse", json=userdata, status=200
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
value = instance.field_from_activity(
- ['https://example.com/user/mouse', 'bleh']
+ ["https://example.com/user/mouse", "bleh"]
)
self.assertIsInstance(value, list)
self.assertEqual(len(value), 1)
self.assertIsInstance(value[0], User)
def test_tag_field(self):
- ''' a special type of many to many field '''
- instance = fields.TagField('User')
+ """ a special type of many to many field """
+ instance = fields.TagField("User")
Serializable = namedtuple(
- 'Serializable',
- ('to_activity', 'remote_id', 'name_field', 'name')
+ "Serializable", ("to_activity", "remote_id", "name_field", "name")
)
- Queryset = namedtuple('Queryset', ('all', 'instance'))
- item = Serializable(
- lambda: {'a': 'b'}, 'https://e.b/c', 'name', 'Name')
- another_item = Serializable(
- lambda: {}, 'example.com', '', '')
+ Queryset = namedtuple("Queryset", ("all", "instance"))
+ item = Serializable(lambda: {"a": "b"}, "https://e.b/c", "name", "Name")
+ another_item = Serializable(lambda: {}, "example.com", "", "")
items = Queryset(lambda: [item], another_item)
result = instance.field_to_activity(items)
self.assertIsInstance(result, list)
self.assertEqual(len(result), 1)
- self.assertEqual(result[0].href, 'https://e.b/c')
- self.assertEqual(result[0].name, 'Name')
- self.assertEqual(result[0].type, 'Serializable')
-
+ self.assertEqual(result[0].href, "https://e.b/c")
+ self.assertEqual(result[0].name, "Name")
+ self.assertEqual(result[0].type, "Serializable")
def test_tag_field_from_activity(self):
- ''' loadin' a list of items from Links '''
+ """ loadin' a list of items from Links """
# TODO
-
@responses.activate
- @patch('bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast')
+ @patch("bookwyrm.models.activitypub_mixin.ObjectMixin.broadcast")
def test_image_field(self, _):
- ''' storing images '''
+ """ storing images """
user = User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
image_file = pathlib.Path(__file__).parent.joinpath(
- '../../static/images/default_avi.jpg')
+ "../../static/images/default_avi.jpg"
+ )
image = Image.open(image_file)
output = BytesIO()
image.save(output, format=image.format)
- user.avatar.save(
- 'test.jpg',
- ContentFile(output.getvalue())
- )
+ user.avatar.save("test.jpg", ContentFile(output.getvalue()))
- output = fields.image_serializer(user.avatar, alt='alt text')
+ output = fields.image_serializer(user.avatar, alt="alt text")
self.assertIsNotNone(
re.match(
- r'.*\.jpg',
+ r".*\.jpg",
output.url,
)
)
- self.assertEqual(output.name, 'alt text')
- self.assertEqual(output.type, 'Image')
+ self.assertEqual(output.name, "alt text")
+ self.assertEqual(output.type, "Image")
instance = fields.ImageField()
@@ -433,36 +413,30 @@ class ActivitypubFields(TestCase):
responses.add(
responses.GET,
- 'http://www.example.com/image.jpg',
+ "http://www.example.com/image.jpg",
body=user.avatar.file.read(),
- status=200)
- loaded_image = instance.field_from_activity(
- 'http://www.example.com/image.jpg')
+ status=200,
+ )
+ loaded_image = instance.field_from_activity("http://www.example.com/image.jpg")
self.assertIsInstance(loaded_image, list)
self.assertIsInstance(loaded_image[1], ContentFile)
-
def test_datetime_field(self):
- ''' this one is pretty simple, it just has to use isoformat '''
+ """ this one is pretty simple, it just has to use isoformat """
instance = fields.DateTimeField()
now = timezone.now()
self.assertEqual(instance.field_to_activity(now), now.isoformat())
- self.assertEqual(
- instance.field_from_activity(now.isoformat()), now
- )
- self.assertEqual(instance.field_from_activity('bip'), None)
-
+ self.assertEqual(instance.field_from_activity(now.isoformat()), now)
+ self.assertEqual(instance.field_from_activity("bip"), None)
def test_array_field(self):
- ''' idk why it makes them strings but probably for a good reason '''
+ """ idk why it makes them strings but probably for a good reason """
instance = fields.ArrayField(fields.IntegerField)
- self.assertEqual(instance.field_to_activity([0, 1]), ['0', '1'])
-
+ self.assertEqual(instance.field_to_activity([0, 1]), ["0", "1"])
def test_html_field(self):
- ''' sanitizes html, the sanitizer has its own tests '''
+ """ sanitizes html, the sanitizer has its own tests """
instance = fields.HtmlField()
self.assertEqual(
- instance.field_from_activity('
hi
'),
- '
hi
'
+ instance.field_from_activity("
hi
"), "
hi
"
)
diff --git a/bookwyrm/tests/models/test_import_model.py b/bookwyrm/tests/models/test_import_model.py
index 7ec4e700..38c3b1ed 100644
--- a/bookwyrm/tests/models/test_import_model.py
+++ b/bookwyrm/tests/models/test_import_model.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
import datetime
import json
import pathlib
@@ -14,165 +14,166 @@ from bookwyrm.connectors.abstract_connector import SearchResult
class ImportJob(TestCase):
- ''' this is a fancy one!!! '''
+ """ this is a fancy one!!! """
+
def setUp(self):
- ''' data is from a goodreads export of The Raven Tower '''
+ """ data is from a goodreads export of The Raven Tower """
read_data = {
- 'Book Id': 39395857,
- 'Title': 'The Raven Tower',
- 'Author': 'Ann Leckie',
- 'Author l-f': 'Leckie, Ann',
- 'Additional Authors': '',
- 'ISBN': '="0356506991"',
- 'ISBN13': '="9780356506999"',
- 'My Rating': 0,
- 'Average Rating': 4.06,
- 'Publisher': 'Orbit',
- 'Binding': 'Hardcover',
- 'Number of Pages': 416,
- 'Year Published': 2019,
- 'Original Publication Year': 2019,
- 'Date Read': '2019/04/12',
- 'Date Added': '2019/04/09',
- 'Bookshelves': '',
- 'Bookshelves with positions': '',
- 'Exclusive Shelf': 'read',
- 'My Review': '',
- 'Spoiler': '',
- 'Private Notes': '',
- 'Read Count': 1,
- 'Recommended For': '',
- 'Recommended By': '',
- 'Owned Copies': 0,
- 'Original Purchase Date': '',
- 'Original Purchase Location': '',
- 'Condition': '',
- 'Condition Description': '',
- 'BCID': ''
+ "Book Id": 39395857,
+ "Title": "The Raven Tower",
+ "Author": "Ann Leckie",
+ "Author l-f": "Leckie, Ann",
+ "Additional Authors": "",
+ "ISBN": '="0356506991"',
+ "ISBN13": '="9780356506999"',
+ "My Rating": 0,
+ "Average Rating": 4.06,
+ "Publisher": "Orbit",
+ "Binding": "Hardcover",
+ "Number of Pages": 416,
+ "Year Published": 2019,
+ "Original Publication Year": 2019,
+ "Date Read": "2019/04/12",
+ "Date Added": "2019/04/09",
+ "Bookshelves": "",
+ "Bookshelves with positions": "",
+ "Exclusive Shelf": "read",
+ "My Review": "",
+ "Spoiler": "",
+ "Private Notes": "",
+ "Read Count": 1,
+ "Recommended For": "",
+ "Recommended By": "",
+ "Owned Copies": 0,
+ "Original Purchase Date": "",
+ "Original Purchase Location": "",
+ "Condition": "",
+ "Condition Description": "",
+ "BCID": "",
}
currently_reading_data = read_data.copy()
- currently_reading_data['Exclusive Shelf'] = 'currently-reading'
- currently_reading_data['Date Read'] = ''
+ currently_reading_data["Exclusive Shelf"] = "currently-reading"
+ currently_reading_data["Date Read"] = ""
unknown_read_data = currently_reading_data.copy()
- unknown_read_data['Exclusive Shelf'] = 'read'
- unknown_read_data['Date Read'] = ''
+ unknown_read_data["Exclusive Shelf"] = "read"
+ unknown_read_data["Date Read"] = ""
user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
job = models.ImportJob.objects.create(user=user)
self.item_1 = models.ImportItem.objects.create(
- job=job, index=1, data=currently_reading_data)
- self.item_2 = models.ImportItem.objects.create(
- job=job, index=2, data=read_data)
+ job=job, index=1, data=currently_reading_data
+ )
+ self.item_2 = models.ImportItem.objects.create(job=job, index=2, data=read_data)
self.item_3 = models.ImportItem.objects.create(
- job=job, index=3, data=unknown_read_data)
-
+ job=job, index=3, data=unknown_read_data
+ )
def test_isbn(self):
- ''' it unquotes the isbn13 field from data '''
- expected = '9780356506999'
+ """ it unquotes the isbn13 field from data """
+ expected = "9780356506999"
item = models.ImportItem.objects.get(index=1)
self.assertEqual(item.isbn, expected)
-
def test_shelf(self):
- ''' converts to the local shelf typology '''
- expected = 'reading'
+ """ converts to the local shelf typology """
+ expected = "reading"
self.assertEqual(self.item_1.shelf, expected)
-
def test_date_added(self):
- ''' converts to the local shelf typology '''
+ """ converts to the local shelf typology """
expected = datetime.datetime(2019, 4, 9, 0, 0, tzinfo=timezone.utc)
item = models.ImportItem.objects.get(index=1)
self.assertEqual(item.date_added, expected)
-
def test_date_read(self):
- ''' converts to the local shelf typology '''
+ """ converts to the local shelf typology """
expected = datetime.datetime(2019, 4, 12, 0, 0, tzinfo=timezone.utc)
item = models.ImportItem.objects.get(index=2)
self.assertEqual(item.date_read, expected)
-
def test_currently_reading_reads(self):
- ''' infer currently reading dates where available '''
- expected = [models.ReadThrough(
- start_date=datetime.datetime(2019, 4, 9, 0, 0, tzinfo=timezone.utc)
- )]
+ """ infer currently reading dates where available """
+ expected = [
+ models.ReadThrough(
+ start_date=datetime.datetime(2019, 4, 9, 0, 0, tzinfo=timezone.utc)
+ )
+ ]
actual = models.ImportItem.objects.get(index=1)
self.assertEqual(actual.reads[0].start_date, expected[0].start_date)
self.assertEqual(actual.reads[0].finish_date, expected[0].finish_date)
def test_read_reads(self):
- ''' infer read dates where available '''
+ """ infer read dates where available """
actual = self.item_2
self.assertEqual(
actual.reads[0].start_date,
- datetime.datetime(2019, 4, 9, 0, 0, tzinfo=timezone.utc))
+ datetime.datetime(2019, 4, 9, 0, 0, tzinfo=timezone.utc),
+ )
self.assertEqual(
actual.reads[0].finish_date,
- datetime.datetime(2019, 4, 12, 0, 0, tzinfo=timezone.utc))
+ datetime.datetime(2019, 4, 12, 0, 0, tzinfo=timezone.utc),
+ )
def test_unread_reads(self):
- ''' handle books with no read dates '''
+ """ handle books with no read dates """
expected = []
actual = models.ImportItem.objects.get(index=3)
self.assertEqual(actual.reads, expected)
-
@responses.activate
def test_get_book_from_isbn(self):
- ''' search and load books by isbn (9780356506999) '''
+ """ search and load books by isbn (9780356506999) """
connector_info = models.Connector.objects.create(
- identifier='openlibrary.org',
- name='OpenLibrary',
- connector_file='openlibrary',
- base_url='https://openlibrary.org',
- books_url='https://openlibrary.org',
- covers_url='https://covers.openlibrary.org',
- search_url='https://openlibrary.org/search?q=',
+ identifier="openlibrary.org",
+ name="OpenLibrary",
+ connector_file="openlibrary",
+ base_url="https://openlibrary.org",
+ books_url="https://openlibrary.org",
+ covers_url="https://covers.openlibrary.org",
+ search_url="https://openlibrary.org/search?q=",
priority=3,
)
connector = connector_manager.load_connector(connector_info)
result = SearchResult(
- title='Test Result',
- key='https://openlibrary.org/works/OL1234W',
- author='An Author',
- year='1980',
+ title="Test Result",
+ key="https://openlibrary.org/works/OL1234W",
+ author="An Author",
+ year="1980",
connector=connector,
)
-
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ol_edition.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ol_edition.json")
bookdata = json.loads(datafile.read_bytes())
responses.add(
responses.GET,
- 'https://openlibrary.org/works/OL1234W',
+ "https://openlibrary.org/works/OL1234W",
json=bookdata,
- status=200)
+ status=200,
+ )
responses.add(
responses.GET,
- 'https://openlibrary.org/works/OL15832982W',
+ "https://openlibrary.org/works/OL15832982W",
json=bookdata,
- status=200)
+ status=200,
+ )
responses.add(
responses.GET,
- 'https://openlibrary.org/authors/OL382982A',
- json={'name': 'test author'},
- status=200)
+ "https://openlibrary.org/authors/OL382982A",
+ json={"name": "test author"},
+ status=200,
+ )
- with patch(
- 'bookwyrm.connectors.abstract_connector.load_more_data.delay'):
+ with patch("bookwyrm.connectors.abstract_connector.load_more_data.delay"):
with patch(
- 'bookwyrm.connectors.connector_manager.first_search_result'
- ) as search:
+ "bookwyrm.connectors.connector_manager.first_search_result"
+ ) as search:
search.return_value = result
- with patch('bookwyrm.connectors.openlibrary.Connector.' \
- 'get_authors_from_data'):
+ with patch(
+ "bookwyrm.connectors.openlibrary.Connector." "get_authors_from_data"
+ ):
book = self.item_1.get_book_from_isbn()
- self.assertEqual(book.title, 'Sabriel')
+ self.assertEqual(book.title, "Sabriel")
diff --git a/bookwyrm/tests/models/test_list.py b/bookwyrm/tests/models/test_list.py
index d41b81c7..6bc4b796 100644
--- a/bookwyrm/tests/models/test_list.py
+++ b/bookwyrm/tests/models/test_list.py
@@ -1,43 +1,41 @@
-''' testing models '''
+""" testing models """
from unittest.mock import patch
from django.test import TestCase
from bookwyrm import models, settings
-@patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay')
+@patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay")
class List(TestCase):
- ''' some activitypub oddness ahead '''
+ """ some activitypub oddness ahead """
+
def setUp(self):
- ''' look, a list '''
+ """ look, a list """
self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- self.list = models.List.objects.create(
- name='Test List', user=self.user)
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ self.list = models.List.objects.create(name="Test List", user=self.user)
def test_remote_id(self, _):
- ''' shelves use custom remote ids '''
- expected_id = 'https://%s/list/%d' % \
- (settings.DOMAIN, self.list.id)
+ """ shelves use custom remote ids """
+ expected_id = "https://%s/list/%d" % (settings.DOMAIN, self.list.id)
self.assertEqual(self.list.get_remote_id(), expected_id)
-
def test_to_activity(self, _):
- ''' jsonify it '''
+ """ jsonify it """
activity_json = self.list.to_activity()
self.assertIsInstance(activity_json, dict)
- self.assertEqual(activity_json['id'], self.list.remote_id)
- self.assertEqual(activity_json['totalItems'], 0)
- self.assertEqual(activity_json['type'], 'BookList')
- self.assertEqual(activity_json['name'], 'Test List')
- self.assertEqual(activity_json['owner'], self.user.remote_id)
+ self.assertEqual(activity_json["id"], self.list.remote_id)
+ self.assertEqual(activity_json["totalItems"], 0)
+ self.assertEqual(activity_json["type"], "BookList")
+ self.assertEqual(activity_json["name"], "Test List")
+ self.assertEqual(activity_json["owner"], self.user.remote_id)
def test_list_item(self, _):
- ''' a list entry '''
- work = models.Work.objects.create(title='hello')
- book = models.Edition.objects.create(title='hi', parent_work=work)
+ """ a list entry """
+ work = models.Work.objects.create(title="hello")
+ book = models.Edition.objects.create(title="hi", parent_work=work)
item = models.ListItem.objects.create(
book_list=self.list,
book=book,
@@ -47,11 +45,11 @@ class List(TestCase):
self.assertTrue(item.approved)
add_activity = item.to_add_activity()
- self.assertEqual(add_activity['actor'], self.user.remote_id)
- self.assertEqual(add_activity['object']['id'], book.remote_id)
- self.assertEqual(add_activity['target'], self.list.remote_id)
+ self.assertEqual(add_activity["actor"], self.user.remote_id)
+ self.assertEqual(add_activity["object"]["id"], book.remote_id)
+ self.assertEqual(add_activity["target"], self.list.remote_id)
remove_activity = item.to_remove_activity()
- self.assertEqual(remove_activity['actor'], self.user.remote_id)
- self.assertEqual(remove_activity['object']['id'], book.remote_id)
- self.assertEqual(remove_activity['target'], self.list.remote_id)
+ self.assertEqual(remove_activity["actor"], self.user.remote_id)
+ self.assertEqual(remove_activity["object"]["id"], book.remote_id)
+ self.assertEqual(remove_activity["target"], self.list.remote_id)
diff --git a/bookwyrm/tests/models/test_readthrough_model.py b/bookwyrm/tests/models/test_readthrough_model.py
index 3fcdf1e4..f69e8779 100644
--- a/bookwyrm/tests/models/test_readthrough_model.py
+++ b/bookwyrm/tests/models/test_readthrough_model.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from django.test import TestCase
from django.core.exceptions import ValidationError
@@ -6,39 +6,36 @@ from bookwyrm import models, settings
class ReadThrough(TestCase):
- ''' some activitypub oddness ahead '''
- def setUp(self):
- ''' look, a shelf '''
- self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
+ """ some activitypub oddness ahead """
- self.work = models.Work.objects.create(
- title='Example Work'
+ def setUp(self):
+ """ look, a shelf """
+ self.user = models.User.objects.create_user(
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
)
+ self.work = models.Work.objects.create(title="Example Work")
+
self.edition = models.Edition.objects.create(
- title='Example Edition',
- parent_work=self.work
+ title="Example Edition", parent_work=self.work
)
self.work.default_edition = self.edition
self.work.save()
self.readthrough = models.ReadThrough.objects.create(
- user=self.user,
- book=self.edition)
+ user=self.user, book=self.edition
+ )
def test_progress_update(self):
- ''' Test progress updates '''
- self.readthrough.create_update() # No-op, no progress yet
+ """ Test progress updates """
+ self.readthrough.create_update() # No-op, no progress yet
self.readthrough.progress = 10
self.readthrough.create_update()
self.readthrough.progress = 20
self.readthrough.progress_mode = models.ProgressMode.PERCENT
self.readthrough.create_update()
- updates = self.readthrough.progressupdate_set \
- .order_by('created_date').all()
+ updates = self.readthrough.progressupdate_set.order_by("created_date").all()
self.assertEqual(len(updates), 2)
self.assertEqual(updates[0].progress, 10)
self.assertEqual(updates[0].mode, models.ProgressMode.PAGE)
diff --git a/bookwyrm/tests/models/test_relationship_models.py b/bookwyrm/tests/models/test_relationship_models.py
index 0ef53450..0e842b21 100644
--- a/bookwyrm/tests/models/test_relationship_models.py
+++ b/bookwyrm/tests/models/test_relationship_models.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from unittest.mock import patch
from django.test import TestCase
@@ -6,87 +6,79 @@ from bookwyrm import models
class Relationship(TestCase):
- ''' following, blocking, stuff like that '''
+ """ following, blocking, stuff like that """
+
def setUp(self):
- ''' we need some users for this '''
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ """ we need some users for this """
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ 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, localname='mouse')
- self.local_user.remote_id = 'http://local.com/user/mouse'
+ "mouse", "mouse@mouse.com", "mouseword", local=True, localname="mouse"
+ )
+ self.local_user.remote_id = "http://local.com/user/mouse"
self.local_user.save(broadcast=False)
-
def test_user_follows_from_request(self):
- ''' convert a follow request into a follow '''
+ """ convert a follow request into a follow """
real_broadcast = models.UserFollowRequest.broadcast
+
def mock_broadcast(_, activity, user):
- ''' introspect what's being sent out '''
+ """ introspect what's being sent out """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Follow')
+ self.assertEqual(activity["type"], "Follow")
models.UserFollowRequest.broadcast = mock_broadcast
request = models.UserFollowRequest.objects.create(
- user_subject=self.local_user,
- user_object=self.remote_user
+ user_subject=self.local_user, user_object=self.remote_user
)
self.assertEqual(
- request.remote_id,
- 'http://local.com/user/mouse#follows/%d' % request.id
+ request.remote_id, "http://local.com/user/mouse#follows/%d" % request.id
)
- self.assertEqual(request.status, 'follow_request')
+ self.assertEqual(request.status, "follow_request")
rel = models.UserFollows.from_request(request)
self.assertEqual(
- rel.remote_id,
- 'http://local.com/user/mouse#follows/%d' % request.id
+ rel.remote_id, "http://local.com/user/mouse#follows/%d" % request.id
)
- self.assertEqual(rel.status, 'follows')
+ self.assertEqual(rel.status, "follows")
self.assertEqual(rel.user_subject, self.local_user)
self.assertEqual(rel.user_object, self.remote_user)
models.UserFollowRequest.broadcast = real_broadcast
-
-
def test_user_follows_from_request_custom_remote_id(self):
- ''' store a specific remote id for a relationship provided by remote '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ store a specific remote id for a relationship provided by remote """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
request = models.UserFollowRequest.objects.create(
user_subject=self.local_user,
user_object=self.remote_user,
- remote_id='http://antoher.server/sdkfhskdjf/23'
+ remote_id="http://antoher.server/sdkfhskdjf/23",
)
- self.assertEqual(
- request.remote_id,
- 'http://antoher.server/sdkfhskdjf/23'
- )
- self.assertEqual(request.status, 'follow_request')
+ self.assertEqual(request.remote_id, "http://antoher.server/sdkfhskdjf/23")
+ self.assertEqual(request.status, "follow_request")
rel = models.UserFollows.from_request(request)
- self.assertEqual(
- rel.remote_id,
- 'http://antoher.server/sdkfhskdjf/23'
- )
- self.assertEqual(rel.status, 'follows')
+ self.assertEqual(rel.remote_id, "http://antoher.server/sdkfhskdjf/23")
+ self.assertEqual(rel.status, "follows")
self.assertEqual(rel.user_subject, self.local_user)
self.assertEqual(rel.user_object, self.remote_user)
-
def test_follow_request_activity(self):
- ''' accept a request and make it a relationship '''
+ """ accept a request and make it a relationship """
real_broadcast = models.UserFollowRequest.broadcast
+
def mock_broadcast(_, activity, user):
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object'], self.remote_user.remote_id)
- self.assertEqual(activity['type'], 'Follow')
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"], self.remote_user.remote_id)
+ self.assertEqual(activity["type"], "Follow")
models.UserFollowRequest.broadcast = mock_broadcast
models.UserFollowRequest.objects.create(
@@ -95,15 +87,15 @@ class Relationship(TestCase):
)
models.UserFollowRequest.broadcast = real_broadcast
-
def test_follow_request_accept(self):
- ''' accept a request and make it a relationship '''
+ """ accept a request and make it a relationship """
real_broadcast = models.UserFollowRequest.broadcast
+
def mock_broadcast(_, activity, user):
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Accept')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['id'], 'https://www.hi.com/')
+ self.assertEqual(activity["type"], "Accept")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["id"], "https://www.hi.com/")
self.local_user.manually_approves_followers = True
self.local_user.save(broadcast=False)
@@ -111,7 +103,7 @@ class Relationship(TestCase):
request = models.UserFollowRequest.objects.create(
user_subject=self.remote_user,
user_object=self.local_user,
- remote_id='https://www.hi.com/'
+ remote_id="https://www.hi.com/",
)
request.accept()
@@ -122,16 +114,15 @@ class Relationship(TestCase):
self.assertEqual(rel.user_object, self.local_user)
models.UserFollowRequest.broadcast = real_broadcast
-
def test_follow_request_reject(self):
- ''' accept a request and make it a relationship '''
+ """ accept a request and make it a relationship """
real_broadcast = models.UserFollowRequest.broadcast
+
def mock_reject(_, activity, user):
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Reject')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(
- activity['object']['id'], request.remote_id)
+ self.assertEqual(activity["type"], "Reject")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["id"], request.remote_id)
models.UserFollowRequest.broadcast = mock_reject
self.local_user.manually_approves_followers = True
diff --git a/bookwyrm/tests/models/test_shelf_model.py b/bookwyrm/tests/models/test_shelf_model.py
index c997326e..3bbb9890 100644
--- a/bookwyrm/tests/models/test_shelf_model.py
+++ b/bookwyrm/tests/models/test_shelf_model.py
@@ -1,116 +1,120 @@
-''' testing models '''
+""" testing models """
from django.test import TestCase
from bookwyrm import models, settings
-#pylint: disable=unused-argument
+# pylint: disable=unused-argument
class Shelf(TestCase):
- ''' some activitypub oddness ahead '''
+ """ some activitypub oddness ahead """
+
def setUp(self):
- ''' look, a shelf '''
+ """ look, a shelf """
self.local_user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- work = models.Work.objects.create(title='Test Work')
- self.book = models.Edition.objects.create(
- title='test book',
- parent_work=work)
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
+ work = models.Work.objects.create(title="Test Work")
+ self.book = models.Edition.objects.create(title="test book", parent_work=work)
def test_remote_id(self):
- ''' shelves use custom remote ids '''
+ """ shelves use custom remote ids """
real_broadcast = models.Shelf.broadcast
+
def broadcast_mock(_, activity, user, **kwargs):
- ''' nah '''
+ """ nah """
+
models.Shelf.broadcast = broadcast_mock
shelf = models.Shelf.objects.create(
- name='Test Shelf', identifier='test-shelf',
- user=self.local_user)
- expected_id = 'https://%s/user/mouse/shelf/test-shelf' % settings.DOMAIN
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
+ )
+ expected_id = "https://%s/user/mouse/shelf/test-shelf" % settings.DOMAIN
self.assertEqual(shelf.get_remote_id(), expected_id)
models.Shelf.broadcast = real_broadcast
-
def test_to_activity(self):
- ''' jsonify it '''
+ """ jsonify it """
real_broadcast = models.Shelf.broadcast
+
def empty_mock(_, activity, user, **kwargs):
- ''' nah '''
+ """ nah """
+
models.Shelf.broadcast = empty_mock
shelf = models.Shelf.objects.create(
- name='Test Shelf', identifier='test-shelf',
- user=self.local_user)
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
+ )
activity_json = shelf.to_activity()
self.assertIsInstance(activity_json, dict)
- self.assertEqual(activity_json['id'], shelf.remote_id)
- self.assertEqual(activity_json['totalItems'], 0)
- self.assertEqual(activity_json['type'], 'Shelf')
- self.assertEqual(activity_json['name'], 'Test Shelf')
- self.assertEqual(activity_json['owner'], self.local_user.remote_id)
+ self.assertEqual(activity_json["id"], shelf.remote_id)
+ self.assertEqual(activity_json["totalItems"], 0)
+ self.assertEqual(activity_json["type"], "Shelf")
+ self.assertEqual(activity_json["name"], "Test Shelf")
+ self.assertEqual(activity_json["owner"], self.local_user.remote_id)
models.Shelf.broadcast = real_broadcast
-
def test_create_update_shelf(self):
- ''' create and broadcast shelf creation '''
+ """ create and broadcast shelf creation """
real_broadcast = models.Shelf.broadcast
+
def create_mock(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Create')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['name'], 'Test Shelf')
+ self.assertEqual(activity["type"], "Create")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["name"], "Test Shelf")
+
models.Shelf.broadcast = create_mock
shelf = models.Shelf.objects.create(
- name='Test Shelf', identifier='test-shelf', user=self.local_user)
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
+ )
def update_mock(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Update')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['name'], 'arthur russel')
+ self.assertEqual(activity["type"], "Update")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["name"], "arthur russel")
+
models.Shelf.broadcast = update_mock
- shelf.name = 'arthur russel'
+ shelf.name = "arthur russel"
shelf.save()
- self.assertEqual(shelf.name, 'arthur russel')
+ self.assertEqual(shelf.name, "arthur russel")
models.Shelf.broadcast = real_broadcast
-
def test_shelve(self):
- ''' create and broadcast shelf creation '''
+ """ create and broadcast shelf creation """
real_broadcast = models.Shelf.broadcast
real_shelfbook_broadcast = models.ShelfBook.broadcast
def add_mock(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['id'], self.book.remote_id)
- self.assertEqual(activity['target'], shelf.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["id"], self.book.remote_id)
+ self.assertEqual(activity["target"], shelf.remote_id)
def remove_mock(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Remove')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['id'], self.book.remote_id)
- self.assertEqual(activity['target'], shelf.remote_id)
+ self.assertEqual(activity["type"], "Remove")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["id"], self.book.remote_id)
+ self.assertEqual(activity["target"], shelf.remote_id)
def empty_mock(_, activity, user, **kwargs):
- ''' nah '''
+ """ nah """
models.Shelf.broadcast = empty_mock
shelf = models.Shelf.objects.create(
- name='Test Shelf', identifier='test-shelf', user=self.local_user)
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
+ )
models.ShelfBook.broadcast = add_mock
shelf_book = models.ShelfBook.objects.create(
- shelf=shelf,
- user=self.local_user,
- book=self.book)
+ shelf=shelf, user=self.local_user, book=self.book
+ )
self.assertEqual(shelf.books.first(), self.book)
models.ShelfBook.broadcast = remove_mock
diff --git a/bookwyrm/tests/models/test_status_model.py b/bookwyrm/tests/models/test_status_model.py
index c6911b6d..21982c20 100644
--- a/bookwyrm/tests/models/test_status_model.py
+++ b/bookwyrm/tests/models/test_status_model.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from unittest.mock import patch
from io import BytesIO
import pathlib
@@ -12,44 +12,45 @@ from django.utils import timezone
from bookwyrm import models, settings
-@patch('bookwyrm.models.Status.broadcast')
+@patch("bookwyrm.models.Status.broadcast")
class Status(TestCase):
- ''' lotta types of statuses '''
+ """ lotta types of statuses """
+
def setUp(self):
- ''' useful things for creating a status '''
+ """ useful things for creating a status """
self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- self.book = models.Edition.objects.create(title='Test Edition')
+ "mouse", "mouse@mouse.mouse", "mouseword", local=True, localname="mouse"
+ )
+ self.book = models.Edition.objects.create(title="Test Edition")
image_file = pathlib.Path(__file__).parent.joinpath(
- '../../static/images/default_avi.jpg')
+ "../../static/images/default_avi.jpg"
+ )
image = Image.open(image_file)
output = BytesIO()
- with patch('bookwyrm.models.Status.broadcast'):
+ with patch("bookwyrm.models.Status.broadcast"):
image.save(output, format=image.format)
- self.book.cover.save(
- 'test.jpg',
- ContentFile(output.getvalue())
- )
+ self.book.cover.save("test.jpg", ContentFile(output.getvalue()))
def test_status_generated_fields(self, _):
- ''' setting remote id '''
- status = models.Status.objects.create(content='bleh', user=self.user)
- expected_id = 'https://%s/user/mouse/status/%d' % \
- (settings.DOMAIN, status.id)
+ """ setting remote id """
+ status = models.Status.objects.create(content="bleh", user=self.user)
+ expected_id = "https://%s/user/mouse/status/%d" % (settings.DOMAIN, status.id)
self.assertEqual(status.remote_id, expected_id)
- self.assertEqual(status.privacy, 'public')
+ self.assertEqual(status.privacy, "public")
def test_replies(self, _):
- ''' get a list of replies '''
- parent = models.Status.objects.create(content='hi', user=self.user)
+ """ get a list of replies """
+ parent = models.Status.objects.create(content="hi", user=self.user)
child = models.Status.objects.create(
- content='hello', reply_parent=parent, user=self.user)
+ content="hello", reply_parent=parent, user=self.user
+ )
models.Review.objects.create(
- content='hey', reply_parent=parent, user=self.user, book=self.book)
+ content="hey", reply_parent=parent, user=self.user, book=self.book
+ )
models.Status.objects.create(
- content='hi hello', reply_parent=child, user=self.user)
+ content="hi hello", reply_parent=child, user=self.user
+ )
replies = models.Status.replies(parent)
self.assertEqual(replies.count(), 2)
@@ -58,199 +59,228 @@ class Status(TestCase):
self.assertIsInstance(replies.last(), models.Review)
def test_status_type(self, _):
- ''' class name '''
- self.assertEqual(models.Status().status_type, 'Note')
- self.assertEqual(models.Review().status_type, 'Review')
- self.assertEqual(models.Quotation().status_type, 'Quotation')
- self.assertEqual(models.Comment().status_type, 'Comment')
- self.assertEqual(models.Boost().status_type, 'Announce')
+ """ class name """
+ self.assertEqual(models.Status().status_type, "Note")
+ self.assertEqual(models.Review().status_type, "Review")
+ self.assertEqual(models.Quotation().status_type, "Quotation")
+ self.assertEqual(models.Comment().status_type, "Comment")
+ self.assertEqual(models.Boost().status_type, "Announce")
def test_boostable(self, _):
- ''' can a status be boosted, based on privacy '''
- self.assertTrue(models.Status(privacy='public').boostable)
- self.assertTrue(models.Status(privacy='unlisted').boostable)
- self.assertFalse(models.Status(privacy='followers').boostable)
- self.assertFalse(models.Status(privacy='direct').boostable)
+ """ can a status be boosted, based on privacy """
+ self.assertTrue(models.Status(privacy="public").boostable)
+ self.assertTrue(models.Status(privacy="unlisted").boostable)
+ self.assertFalse(models.Status(privacy="followers").boostable)
+ self.assertFalse(models.Status(privacy="direct").boostable)
def test_to_replies(self, _):
- ''' activitypub replies collection '''
- parent = models.Status.objects.create(content='hi', user=self.user)
+ """ activitypub replies collection """
+ parent = models.Status.objects.create(content="hi", user=self.user)
child = models.Status.objects.create(
- content='hello', reply_parent=parent, user=self.user)
+ content="hello", reply_parent=parent, user=self.user
+ )
models.Review.objects.create(
- content='hey', reply_parent=parent, user=self.user, book=self.book)
+ content="hey", reply_parent=parent, user=self.user, book=self.book
+ )
models.Status.objects.create(
- content='hi hello', reply_parent=child, user=self.user)
+ content="hi hello", reply_parent=child, user=self.user
+ )
replies = parent.to_replies()
- self.assertEqual(replies['id'], '%s/replies' % parent.remote_id)
- self.assertEqual(replies['totalItems'], 2)
+ self.assertEqual(replies["id"], "%s/replies" % parent.remote_id)
+ self.assertEqual(replies["totalItems"], 2)
def test_status_to_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
- status = models.Status.objects.create(
- content='test content', user=self.user)
+ """ subclass of the base model version with a "pure" serializer """
+ status = models.Status.objects.create(content="test content", user=self.user)
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Note')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['sensitive'], False)
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Note")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["sensitive"], False)
def test_status_to_activity_tombstone(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Status.objects.create(
- content='test content', user=self.user,
- deleted=True, deleted_date=timezone.now())
+ content="test content",
+ user=self.user,
+ deleted=True,
+ deleted_date=timezone.now(),
+ )
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Tombstone')
- self.assertFalse(hasattr(activity, 'content'))
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Tombstone")
+ self.assertFalse(hasattr(activity, "content"))
def test_status_to_pure_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
- status = models.Status.objects.create(
- content='test content', user=self.user)
+ """ subclass of the base model version with a "pure" serializer """
+ status = models.Status.objects.create(content="test content", user=self.user)
activity = status.to_activity(pure=True)
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Note')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['sensitive'], False)
- self.assertEqual(activity['attachment'], [])
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Note")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["sensitive"], False)
+ self.assertEqual(activity["attachment"], [])
def test_generated_note_to_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.GeneratedNote.objects.create(
- content='test content', user=self.user)
+ content="test content", user=self.user
+ )
status.mention_books.set([self.book])
status.mention_users.set([self.user])
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'GeneratedNote')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['sensitive'], False)
- self.assertEqual(len(activity['tag']), 2)
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "GeneratedNote")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["sensitive"], False)
+ self.assertEqual(len(activity["tag"]), 2)
def test_generated_note_to_pure_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.GeneratedNote.objects.create(
- content='test content', user=self.user)
+ content="test content", user=self.user
+ )
status.mention_books.set([self.book])
status.mention_users.set([self.user])
activity = status.to_activity(pure=True)
- self.assertEqual(activity['id'], status.remote_id)
+ self.assertEqual(activity["id"], status.remote_id)
self.assertEqual(
- activity['content'],
- 'mouse test content
"Test Edition" ' % \
- self.book.remote_id)
- self.assertEqual(len(activity['tag']), 2)
- self.assertEqual(activity['type'], 'Note')
- self.assertEqual(activity['sensitive'], False)
- self.assertIsInstance(activity['attachment'], list)
- self.assertEqual(activity['attachment'][0].type, 'Image')
- self.assertEqual(activity['attachment'][0].url, 'https://%s%s' % \
- (settings.DOMAIN, self.book.cover.url))
+ activity["content"],
+ 'mouse test content
"Test Edition" ' % self.book.remote_id,
+ )
+ self.assertEqual(len(activity["tag"]), 2)
+ self.assertEqual(activity["type"], "Note")
+ self.assertEqual(activity["sensitive"], False)
+ self.assertIsInstance(activity["attachment"], list)
+ self.assertEqual(activity["attachment"][0].type, "Image")
self.assertEqual(
- activity['attachment'][0].name, 'Test Edition cover')
+ activity["attachment"][0].url,
+ "https://%s%s" % (settings.DOMAIN, self.book.cover.url),
+ )
+ self.assertEqual(activity["attachment"][0].name, "Test Edition")
def test_comment_to_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Comment.objects.create(
- content='test content', user=self.user, book=self.book)
+ content="test content", user=self.user, book=self.book
+ )
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Comment')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['inReplyToBook'], self.book.remote_id)
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Comment")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["inReplyToBook"], self.book.remote_id)
def test_comment_to_pure_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Comment.objects.create(
- content='test content', user=self.user, book=self.book)
+ content="test content", user=self.user, book=self.book
+ )
activity = status.to_activity(pure=True)
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Note')
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Note")
self.assertEqual(
- activity['content'],
- 'test content
(comment on "Test Edition" )
' %
- self.book.remote_id)
- self.assertEqual(activity['attachment'][0].type, 'Image')
- self.assertEqual(activity['attachment'][0].url, 'https://%s%s' % \
- (settings.DOMAIN, self.book.cover.url))
+ activity["content"],
+ 'test content
(comment on "Test Edition" )
'
+ % self.book.remote_id,
+ )
+ self.assertEqual(activity["attachment"][0].type, "Image")
self.assertEqual(
- activity['attachment'][0].name, 'Test Edition cover')
+ activity["attachment"][0].url,
+ "https://%s%s" % (settings.DOMAIN, self.book.cover.url),
+ )
+ self.assertEqual(activity["attachment"][0].name, "Test Edition")
def test_quotation_to_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Quotation.objects.create(
- quote='a sickening sense', content='test content',
- user=self.user, book=self.book)
+ quote="a sickening sense",
+ content="test content",
+ user=self.user,
+ book=self.book,
+ )
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Quotation')
- self.assertEqual(activity['quote'], 'a sickening sense')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['inReplyToBook'], self.book.remote_id)
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Quotation")
+ self.assertEqual(activity["quote"], "a sickening sense")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["inReplyToBook"], self.book.remote_id)
def test_quotation_to_pure_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Quotation.objects.create(
- quote='a sickening sense', content='test content',
- user=self.user, book=self.book)
+ quote="a sickening sense",
+ content="test content",
+ user=self.user,
+ book=self.book,
+ )
activity = status.to_activity(pure=True)
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Note')
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Note")
self.assertEqual(
- activity['content'],
- 'a sickening sense
-- "Test Edition"
' \
- 'test content' % self.book.remote_id)
- self.assertEqual(activity['attachment'][0].type, 'Image')
- self.assertEqual(activity['attachment'][0].url, 'https://%s%s' % \
- (settings.DOMAIN, self.book.cover.url))
+ activity["content"],
+ 'a sickening sense
-- "Test Edition"
'
+ "test content" % self.book.remote_id,
+ )
+ self.assertEqual(activity["attachment"][0].type, "Image")
self.assertEqual(
- activity['attachment'][0].name, 'Test Edition cover')
+ activity["attachment"][0].url,
+ "https://%s%s" % (settings.DOMAIN, self.book.cover.url),
+ )
+ self.assertEqual(activity["attachment"][0].name, "Test Edition")
def test_review_to_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Review.objects.create(
- name='Review name', content='test content', rating=3,
- user=self.user, book=self.book)
+ name="Review name",
+ content="test content",
+ rating=3,
+ user=self.user,
+ book=self.book,
+ )
activity = status.to_activity()
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Review')
- self.assertEqual(activity['rating'], 3)
- self.assertEqual(activity['name'], 'Review name')
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['inReplyToBook'], self.book.remote_id)
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Review")
+ self.assertEqual(activity["rating"], 3)
+ self.assertEqual(activity["name"], "Review name")
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["inReplyToBook"], self.book.remote_id)
def test_review_to_pure_activity(self, _):
- ''' subclass of the base model version with a "pure" serializer '''
+ """ subclass of the base model version with a "pure" serializer """
status = models.Review.objects.create(
- name='Review name', content='test content', rating=3,
- user=self.user, book=self.book)
+ name="Review name",
+ content="test content",
+ rating=3,
+ user=self.user,
+ book=self.book,
+ )
activity = status.to_activity(pure=True)
- self.assertEqual(activity['id'], status.remote_id)
- self.assertEqual(activity['type'], 'Article')
+ self.assertEqual(activity["id"], status.remote_id)
+ self.assertEqual(activity["type"], "Article")
self.assertEqual(
- activity['name'], 'Review of "%s" (3 stars): Review name' \
- % self.book.title)
- self.assertEqual(activity['content'], 'test content')
- self.assertEqual(activity['attachment'][0].type, 'Image')
- self.assertEqual(activity['attachment'][0].url, 'https://%s%s' % \
- (settings.DOMAIN, self.book.cover.url))
+ activity["name"], 'Review of "%s" (3 stars): Review name' % self.book.title
+ )
+ self.assertEqual(activity["content"], "test content")
+ self.assertEqual(activity["attachment"][0].type, "Image")
self.assertEqual(
- activity['attachment'][0].name, 'Test Edition cover')
+ activity["attachment"][0].url,
+ "https://%s%s" % (settings.DOMAIN, self.book.cover.url),
+ )
+ self.assertEqual(activity["attachment"][0].name, "Test Edition")
def test_favorite(self, _):
- ''' fav a status '''
+ """ fav a status """
real_broadcast = models.Favorite.broadcast
+
def fav_broadcast_mock(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.user.remote_id)
- self.assertEqual(activity['type'], 'Like')
+ self.assertEqual(activity["type"], "Like")
+
models.Favorite.broadcast = fav_broadcast_mock
- status = models.Status.objects.create(
- content='test content', user=self.user)
+ status = models.Status.objects.create(content="test content", user=self.user)
fav = models.Favorite.objects.create(status=status, user=self.user)
# can't fav a status twice
@@ -258,50 +288,47 @@ class Status(TestCase):
models.Favorite.objects.create(status=status, user=self.user)
activity = fav.to_activity()
- self.assertEqual(activity['type'], 'Like')
- self.assertEqual(activity['actor'], self.user.remote_id)
- self.assertEqual(activity['object'], status.remote_id)
+ self.assertEqual(activity["type"], "Like")
+ self.assertEqual(activity["actor"], self.user.remote_id)
+ self.assertEqual(activity["object"], status.remote_id)
models.Favorite.broadcast = real_broadcast
def test_boost(self, _):
- ''' boosting, this one's a bit fussy '''
- status = models.Status.objects.create(
- content='test content', user=self.user)
- boost = models.Boost.objects.create(
- boosted_status=status, user=self.user)
+ """ boosting, this one's a bit fussy """
+ status = models.Status.objects.create(content="test content", user=self.user)
+ boost = models.Boost.objects.create(boosted_status=status, user=self.user)
activity = boost.to_activity()
- self.assertEqual(activity['actor'], self.user.remote_id)
- self.assertEqual(activity['object'], status.remote_id)
- self.assertEqual(activity['type'], 'Announce')
+ self.assertEqual(activity["actor"], self.user.remote_id)
+ self.assertEqual(activity["object"], status.remote_id)
+ self.assertEqual(activity["type"], "Announce")
self.assertEqual(activity, boost.to_activity(pure=True))
def test_notification(self, _):
- ''' a simple model '''
+ """ a simple model """
notification = models.Notification.objects.create(
- user=self.user, notification_type='FAVORITE')
+ user=self.user, notification_type="FAVORITE"
+ )
self.assertFalse(notification.read)
with self.assertRaises(IntegrityError):
models.Notification.objects.create(
- user=self.user, notification_type='GLORB')
-
+ user=self.user, notification_type="GLORB"
+ )
def test_create_broadcast(self, broadcast_mock):
- ''' should send out two verions of a status on create '''
- models.Comment.objects.create(
- content='hi', user=self.user, book=self.book)
+ """ should send out two verions of a status on create """
+ models.Comment.objects.create(content="hi", user=self.user, book=self.book)
self.assertEqual(broadcast_mock.call_count, 2)
pure_call = broadcast_mock.call_args_list[0]
bw_call = broadcast_mock.call_args_list[1]
- self.assertEqual(pure_call[1]['software'], 'other')
+ self.assertEqual(pure_call[1]["software"], "other")
args = pure_call[0][0]
- self.assertEqual(args['type'], 'Create')
- self.assertEqual(args['object']['type'], 'Note')
- self.assertTrue('content' in args['object'])
+ self.assertEqual(args["type"], "Create")
+ self.assertEqual(args["object"]["type"], "Note")
+ self.assertTrue("content" in args["object"])
-
- self.assertEqual(bw_call[1]['software'], 'bookwyrm')
+ self.assertEqual(bw_call[1]["software"], "bookwyrm")
args = bw_call[0][0]
- self.assertEqual(args['type'], 'Create')
- self.assertEqual(args['object']['type'], 'Comment')
+ self.assertEqual(args["type"], "Create")
+ self.assertEqual(args["object"]["type"], "Comment")
diff --git a/bookwyrm/tests/models/test_user_model.py b/bookwyrm/tests/models/test_user_model.py
index ed3ad41a..618e4f78 100644
--- a/bookwyrm/tests/models/test_user_model.py
+++ b/bookwyrm/tests/models/test_user_model.py
@@ -1,4 +1,4 @@
-''' testing models '''
+""" testing models """
from unittest.mock import patch
from django.test import TestCase
import responses
@@ -11,75 +11,84 @@ from bookwyrm.settings import DOMAIN
class User(TestCase):
def setUp(self):
self.user = models.User.objects.create_user(
- 'mouse@%s' % DOMAIN, 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse', name='hi', bookwyrm_user=False)
+ "mouse@%s" % DOMAIN,
+ "mouse@mouse.mouse",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ name="hi",
+ bookwyrm_user=False,
+ )
def test_computed_fields(self):
- ''' username instead of id here '''
- expected_id = 'https://%s/user/mouse' % DOMAIN
+ """ username instead of id here """
+ expected_id = "https://%s/user/mouse" % DOMAIN
self.assertEqual(self.user.remote_id, expected_id)
- self.assertEqual(self.user.username, 'mouse@%s' % DOMAIN)
- self.assertEqual(self.user.localname, 'mouse')
- self.assertEqual(self.user.shared_inbox, 'https://%s/inbox' % DOMAIN)
- self.assertEqual(self.user.inbox, '%s/inbox' % expected_id)
- self.assertEqual(self.user.outbox, '%s/outbox' % expected_id)
+ self.assertEqual(self.user.username, "mouse@%s" % DOMAIN)
+ self.assertEqual(self.user.localname, "mouse")
+ self.assertEqual(self.user.shared_inbox, "https://%s/inbox" % DOMAIN)
+ self.assertEqual(self.user.inbox, "%s/inbox" % expected_id)
+ self.assertEqual(self.user.outbox, "%s/outbox" % expected_id)
self.assertIsNotNone(self.user.key_pair.private_key)
self.assertIsNotNone(self.user.key_pair.public_key)
def test_remote_user(self):
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
user = models.User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword', local=False,
- remote_id='https://example.com/dfjkg', bookwyrm_user=False)
- self.assertEqual(user.username, 'rat@example.com')
-
+ "rat",
+ "rat@rat.rat",
+ "ratword",
+ local=False,
+ remote_id="https://example.com/dfjkg",
+ bookwyrm_user=False,
+ )
+ self.assertEqual(user.username, "rat@example.com")
def test_user_shelves(self):
shelves = models.Shelf.objects.filter(user=self.user).all()
self.assertEqual(len(shelves), 3)
names = [s.name for s in shelves]
- self.assertTrue('To Read' in names)
- self.assertTrue('Currently Reading' in names)
- self.assertTrue('Read' in names)
+ self.assertTrue("To Read" in names)
+ self.assertTrue("Currently Reading" in names)
+ self.assertTrue("Read" in names)
ids = [s.identifier for s in shelves]
- self.assertTrue('to-read' in ids)
- self.assertTrue('reading' in ids)
- self.assertTrue('read' in ids)
-
+ self.assertTrue("to-read" in ids)
+ self.assertTrue("reading" in ids)
+ self.assertTrue("read" in ids)
def test_activitypub_serialize(self):
activity = self.user.to_activity()
- self.assertEqual(activity['id'], self.user.remote_id)
- self.assertEqual(activity['@context'], [
- 'https://www.w3.org/ns/activitystreams',
- 'https://w3id.org/security/v1',
- {
- 'manuallyApprovesFollowers': 'as:manuallyApprovesFollowers',
- 'schema': 'http://schema.org#',
- 'PropertyValue': 'schema:PropertyValue',
- 'value': 'schema:value',
- }
- ])
- self.assertEqual(activity['preferredUsername'], self.user.localname)
- self.assertEqual(activity['name'], self.user.name)
- self.assertEqual(activity['inbox'], self.user.inbox)
- self.assertEqual(activity['outbox'], self.user.outbox)
- self.assertEqual(activity['bookwyrmUser'], False)
- self.assertEqual(activity['discoverable'], True)
- self.assertEqual(activity['type'], 'Person')
+ self.assertEqual(activity["id"], self.user.remote_id)
+ self.assertEqual(
+ activity["@context"],
+ [
+ "https://www.w3.org/ns/activitystreams",
+ "https://w3id.org/security/v1",
+ {
+ "manuallyApprovesFollowers": "as:manuallyApprovesFollowers",
+ "schema": "http://schema.org#",
+ "PropertyValue": "schema:PropertyValue",
+ "value": "schema:value",
+ },
+ ],
+ )
+ self.assertEqual(activity["preferredUsername"], self.user.localname)
+ self.assertEqual(activity["name"], self.user.name)
+ self.assertEqual(activity["inbox"], self.user.inbox)
+ self.assertEqual(activity["outbox"], self.user.outbox)
+ self.assertEqual(activity["bookwyrmUser"], False)
+ self.assertEqual(activity["discoverable"], True)
+ self.assertEqual(activity["type"], "Person")
def test_activitypub_outbox(self):
activity = self.user.to_outbox()
- self.assertEqual(activity['type'], 'OrderedCollection')
- self.assertEqual(activity['id'], self.user.outbox)
- self.assertEqual(activity['totalItems'], 0)
-
+ self.assertEqual(activity["type"], "OrderedCollection")
+ self.assertEqual(activity["id"], self.user.outbox)
+ self.assertEqual(activity["totalItems"], 0)
def test_set_remote_server(self):
server = models.FederatedServer.objects.create(
- server_name=DOMAIN,
- application_type='test type',
- application_version=3
+ server_name=DOMAIN, application_type="test type", application_version=3
)
models.user.set_remote_server(self.user.id)
@@ -91,26 +100,24 @@ class User(TestCase):
def test_get_or_create_remote_server(self):
responses.add(
responses.GET,
- 'https://%s/.well-known/nodeinfo' % DOMAIN,
- json={'links': [{'href': 'http://www.example.com'}, {}]}
+ "https://%s/.well-known/nodeinfo" % DOMAIN,
+ json={"links": [{"href": "http://www.example.com"}, {}]},
)
responses.add(
responses.GET,
- 'http://www.example.com',
- json={'software': {'name': 'hi', 'version': '2'}},
+ "http://www.example.com",
+ json={"software": {"name": "hi", "version": "2"}},
)
server = models.user.get_or_create_remote_server(DOMAIN)
self.assertEqual(server.server_name, DOMAIN)
- self.assertEqual(server.application_type, 'hi')
- self.assertEqual(server.application_version, '2')
+ self.assertEqual(server.application_type, "hi")
+ self.assertEqual(server.application_version, "2")
@responses.activate
def test_get_or_create_remote_server_no_wellknown(self):
responses.add(
- responses.GET,
- 'https://%s/.well-known/nodeinfo' % DOMAIN,
- status=404
+ responses.GET, "https://%s/.well-known/nodeinfo" % DOMAIN, status=404
)
server = models.user.get_or_create_remote_server(DOMAIN)
@@ -122,14 +129,10 @@ class User(TestCase):
def test_get_or_create_remote_server_no_links(self):
responses.add(
responses.GET,
- 'https://%s/.well-known/nodeinfo' % DOMAIN,
- json={'links': [{'href': 'http://www.example.com'}, {}]}
- )
- responses.add(
- responses.GET,
- 'http://www.example.com',
- status=404
+ "https://%s/.well-known/nodeinfo" % DOMAIN,
+ json={"links": [{"href": "http://www.example.com"}, {}]},
)
+ responses.add(responses.GET, "http://www.example.com", status=404)
server = models.user.get_or_create_remote_server(DOMAIN)
self.assertEqual(server.server_name, DOMAIN)
@@ -140,14 +143,10 @@ class User(TestCase):
def test_get_or_create_remote_server_unknown_format(self):
responses.add(
responses.GET,
- 'https://%s/.well-known/nodeinfo' % DOMAIN,
- json={'links': [{'href': 'http://www.example.com'}, {}]}
- )
- responses.add(
- responses.GET,
- 'http://www.example.com',
- json={'fish': 'salmon'}
+ "https://%s/.well-known/nodeinfo" % DOMAIN,
+ json={"links": [{"href": "http://www.example.com"}, {}]},
)
+ responses.add(responses.GET, "http://www.example.com", json={"fish": "salmon"})
server = models.user.get_or_create_remote_server(DOMAIN)
self.assertEqual(server.server_name, DOMAIN)
diff --git a/bookwyrm/tests/test_goodreads_import.py b/bookwyrm/tests/test_goodreads_import.py
index aee9afe4..080ccd15 100644
--- a/bookwyrm/tests/test_goodreads_import.py
+++ b/bookwyrm/tests/test_goodreads_import.py
@@ -1,4 +1,4 @@
-''' testing import '''
+""" testing import """
from collections import namedtuple
import csv
import pathlib
@@ -12,125 +12,117 @@ from bookwyrm.goodreads_import import GoodreadsImporter
from bookwyrm import importer
from bookwyrm.settings import DOMAIN
+
class GoodreadsImport(TestCase):
- ''' importing from goodreads csv '''
+ """ importing from goodreads csv """
+
def setUp(self):
self.importer = GoodreadsImporter()
- ''' use a test csv '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- 'data/goodreads.csv')
- self.csv = open(datafile, 'r', encoding=self.importer.encoding)
+ """ use a test csv """
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ self.csv = open(datafile, "r", encoding=self.importer.encoding)
self.user = models.User.objects.create_user(
- 'mouse', 'mouse@mouse.mouse', 'password', local=True)
+ "mouse", "mouse@mouse.mouse", "password", local=True
+ )
models.Connector.objects.create(
identifier=DOMAIN,
- name='Local',
+ name="Local",
local=True,
- connector_file='self_connector',
- base_url='https://%s' % DOMAIN,
- books_url='https://%s/book' % DOMAIN,
- covers_url='https://%s/images/covers' % DOMAIN,
- search_url='https://%s/search?q=' % DOMAIN,
+ connector_file="self_connector",
+ base_url="https://%s" % DOMAIN,
+ books_url="https://%s/book" % DOMAIN,
+ covers_url="https://%s/images/covers" % DOMAIN,
+ search_url="https://%s/search?q=" % DOMAIN,
priority=1,
)
- work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=work,
)
-
def test_create_job(self):
- ''' creates the import job entry and checks csv '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'public')
+ """ creates the import job entry and checks csv """
+ import_job = self.importer.create_job(self.user, self.csv, False, "public")
self.assertEqual(import_job.user, self.user)
self.assertEqual(import_job.include_reviews, False)
- self.assertEqual(import_job.privacy, 'public')
+ self.assertEqual(import_job.privacy, "public")
import_items = models.ImportItem.objects.filter(job=import_job).all()
self.assertEqual(len(import_items), 3)
self.assertEqual(import_items[0].index, 0)
- self.assertEqual(import_items[0].data['Book Id'], '42036538')
+ self.assertEqual(import_items[0].data["Book Id"], "42036538")
self.assertEqual(import_items[1].index, 1)
- self.assertEqual(import_items[1].data['Book Id'], '52691223')
+ self.assertEqual(import_items[1].data["Book Id"], "52691223")
self.assertEqual(import_items[2].index, 2)
- self.assertEqual(import_items[2].data['Book Id'], '28694510')
-
+ self.assertEqual(import_items[2].data["Book Id"], "28694510")
def test_create_retry_job(self):
- ''' trying again with items that didn't import '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'unlisted')
- import_items = models.ImportItem.objects.filter(
- job=import_job
- ).all()[:2]
+ """ trying again with items that didn't import """
+ import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
+ import_items = models.ImportItem.objects.filter(job=import_job).all()[:2]
- retry = self.importer.create_retry_job(
- self.user, import_job, import_items)
+ retry = self.importer.create_retry_job(self.user, import_job, import_items)
self.assertNotEqual(import_job, retry)
self.assertEqual(retry.user, self.user)
self.assertEqual(retry.include_reviews, False)
- self.assertEqual(retry.privacy, 'unlisted')
+ self.assertEqual(retry.privacy, "unlisted")
retry_items = models.ImportItem.objects.filter(job=retry).all()
self.assertEqual(len(retry_items), 2)
self.assertEqual(retry_items[0].index, 0)
- self.assertEqual(retry_items[0].data['Book Id'], '42036538')
+ self.assertEqual(retry_items[0].data["Book Id"], "42036538")
self.assertEqual(retry_items[1].index, 1)
- self.assertEqual(retry_items[1].data['Book Id'], '52691223')
-
+ self.assertEqual(retry_items[1].data["Book Id"], "52691223")
def test_start_import(self):
- ''' begin loading books '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'unlisted')
- MockTask = namedtuple('Task', ('id'))
+ """ begin loading books """
+ import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
+ MockTask = namedtuple("Task", ("id"))
mock_task = MockTask(7)
- with patch('bookwyrm.importer.import_data.delay') as start:
+ with patch("bookwyrm.importer.import_data.delay") as start:
start.return_value = mock_task
self.importer.start_import(import_job)
import_job.refresh_from_db()
- self.assertEqual(import_job.task_id, '7')
-
+ self.assertEqual(import_job.task_id, "7")
@responses.activate
def test_import_data(self):
- ''' resolve entry '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'unlisted')
- book = models.Edition.objects.create(title='Test Book')
+ """ resolve entry """
+ import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
+ book = models.Edition.objects.create(title="Test Book")
with patch(
- 'bookwyrm.models.import_job.ImportItem.get_book_from_isbn'
- ) as resolve:
+ "bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
+ ) as resolve:
resolve.return_value = book
- with patch('bookwyrm.importer.handle_imported_book'):
+ with patch("bookwyrm.importer.handle_imported_book"):
importer.import_data(self.importer.service, import_job.id)
import_item = models.ImportItem.objects.get(job=import_job, index=0)
self.assertEqual(import_item.book.id, book.id)
-
def test_handle_imported_book(self):
- ''' goodreads import added a book, this adds related connections '''
- shelf = self.user.shelf_set.filter(identifier='read').first()
+ """ goodreads import added a book, this adds related connections """
+ shelf = self.user.shelf_set.filter(identifier="read").first()
self.assertIsNone(shelf.books.first())
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/goodreads.csv')
- csv_file = open(datafile, 'r')
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ csv_file = open(datafile, "r")
for index, entry in enumerate(list(csv.DictReader(csv_file))):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
@@ -145,31 +137,30 @@ class GoodreadsImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 10)
self.assertEqual(readthrough.finish_date.day, 25)
-
def test_handle_imported_book_already_shelved(self):
- ''' goodreads import added a book, this adds related connections '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- shelf = self.user.shelf_set.filter(identifier='to-read').first()
- models.ShelfBook.objects.create(
- shelf=shelf, user=self.user, book=self.book)
+ """ goodreads import added a book, this adds related connections """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ shelf = self.user.shelf_set.filter(identifier="to-read").first()
+ models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/goodreads.csv')
- csv_file = open(datafile, 'r')
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ csv_file = open(datafile, "r")
for index, entry in enumerate(list(csv.DictReader(csv_file))):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
- self.assertIsNone(
- self.user.shelf_set.get(identifier='read').books.first())
+ self.assertIsNone(self.user.shelf_set.get(identifier="read").books.first())
readthrough = models.ReadThrough.objects.get(user=self.user)
self.assertEqual(readthrough.book, self.book)
self.assertEqual(readthrough.start_date.year, 2020)
@@ -179,24 +170,26 @@ class GoodreadsImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 10)
self.assertEqual(readthrough.finish_date.day, 25)
-
def test_handle_import_twice(self):
- ''' re-importing books '''
- shelf = self.user.shelf_set.filter(identifier='read').first()
+ """ re-importing books """
+ shelf = self.user.shelf_set.filter(identifier="read").first()
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/goodreads.csv')
- csv_file = open(datafile, 'r')
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ csv_file = open(datafile, "r")
for index, entry in enumerate(list(csv.DictReader(csv_file))):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
@@ -211,42 +204,44 @@ class GoodreadsImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 10)
self.assertEqual(readthrough.finish_date.day, 25)
-
def test_handle_imported_book_review(self):
- ''' goodreads review import '''
+ """ goodreads review import """
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/goodreads.csv')
- csv_file = open(datafile, 'r')
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ csv_file = open(datafile, "r")
entry = list(csv.DictReader(csv_file))[2]
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=0, data=entry, book=self.book)
+ job_id=import_job.id, index=0, data=entry, book=self.book
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, True, 'unlisted')
+ self.importer.service, self.user, import_item, True, "unlisted"
+ )
review = models.Review.objects.get(book=self.book, user=self.user)
- self.assertEqual(review.content, 'mixed feelings')
+ self.assertEqual(review.content, "mixed feelings")
self.assertEqual(review.rating, 2)
self.assertEqual(review.published_date.year, 2019)
self.assertEqual(review.published_date.month, 7)
self.assertEqual(review.published_date.day, 8)
- self.assertEqual(review.privacy, 'unlisted')
-
+ self.assertEqual(review.privacy, "unlisted")
def test_handle_imported_book_reviews_disabled(self):
- ''' goodreads review import '''
+ """ goodreads review import """
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/goodreads.csv')
- csv_file = open(datafile, 'r')
+ datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
+ csv_file = open(datafile, "r")
entry = list(csv.DictReader(csv_file))[2]
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=0, data=entry, book=self.book)
+ job_id=import_job.id, index=0, data=entry, book=self.book
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'unlisted')
- self.assertFalse(models.Review.objects.filter(
- book=self.book, user=self.user
- ).exists())
+ self.importer.service, self.user, import_item, False, "unlisted"
+ )
+ self.assertFalse(
+ models.Review.objects.filter(book=self.book, user=self.user).exists()
+ )
diff --git a/bookwyrm/tests/test_librarything_import.py b/bookwyrm/tests/test_librarything_import.py
index 2623a504..a8e4cfe4 100644
--- a/bookwyrm/tests/test_librarything_import.py
+++ b/bookwyrm/tests/test_librarything_import.py
@@ -1,4 +1,4 @@
-''' testing import '''
+""" testing import """
from collections import namedtuple
import csv
import pathlib
@@ -11,114 +11,110 @@ from bookwyrm import models, importer
from bookwyrm.librarything_import import LibrarythingImporter
from bookwyrm.settings import DOMAIN
+
class LibrarythingImport(TestCase):
- ''' importing from librarything tsv '''
+ """ importing from librarything tsv """
+
def setUp(self):
self.importer = LibrarythingImporter()
- ''' use a test tsv '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- 'data/librarything.tsv')
+ """ use a test tsv """
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
# Librarything generates latin encoded exports...
- self.csv = open(datafile, 'r', encoding=self.importer.encoding)
+ self.csv = open(datafile, "r", encoding=self.importer.encoding)
self.user = models.User.objects.create_user(
- 'mmai', 'mmai@mmai.mmai', 'password', local=True)
+ "mmai", "mmai@mmai.mmai", "password", local=True
+ )
models.Connector.objects.create(
identifier=DOMAIN,
- name='Local',
+ name="Local",
local=True,
- connector_file='self_connector',
- base_url='https://%s' % DOMAIN,
- books_url='https://%s/book' % DOMAIN,
- covers_url='https://%s/images/covers' % DOMAIN,
- search_url='https://%s/search?q=' % DOMAIN,
+ connector_file="self_connector",
+ base_url="https://%s" % DOMAIN,
+ books_url="https://%s/book" % DOMAIN,
+ covers_url="https://%s/images/covers" % DOMAIN,
+ search_url="https://%s/search?q=" % DOMAIN,
priority=1,
)
- work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=work,
)
-
def test_create_job(self):
- ''' creates the import job entry and checks csv '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'public')
+ """ creates the import job entry and checks csv """
+ import_job = self.importer.create_job(self.user, self.csv, False, "public")
self.assertEqual(import_job.user, self.user)
self.assertEqual(import_job.include_reviews, False)
- self.assertEqual(import_job.privacy, 'public')
+ self.assertEqual(import_job.privacy, "public")
import_items = models.ImportItem.objects.filter(job=import_job).all()
self.assertEqual(len(import_items), 3)
self.assertEqual(import_items[0].index, 0)
- self.assertEqual(import_items[0].data['Book Id'], '5498194')
+ self.assertEqual(import_items[0].data["Book Id"], "5498194")
self.assertEqual(import_items[1].index, 1)
- self.assertEqual(import_items[1].data['Book Id'], '5015319')
+ self.assertEqual(import_items[1].data["Book Id"], "5015319")
self.assertEqual(import_items[2].index, 2)
- self.assertEqual(import_items[2].data['Book Id'], '5015399')
-
+ self.assertEqual(import_items[2].data["Book Id"], "5015399")
def test_create_retry_job(self):
- ''' trying again with items that didn't import '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'unlisted')
- import_items = models.ImportItem.objects.filter(
- job=import_job
- ).all()[:2]
+ """ trying again with items that didn't import """
+ import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
+ import_items = models.ImportItem.objects.filter(job=import_job).all()[:2]
- retry = self.importer.create_retry_job(
- self.user, import_job, import_items)
+ retry = self.importer.create_retry_job(self.user, import_job, import_items)
self.assertNotEqual(import_job, retry)
self.assertEqual(retry.user, self.user)
self.assertEqual(retry.include_reviews, False)
- self.assertEqual(retry.privacy, 'unlisted')
+ self.assertEqual(retry.privacy, "unlisted")
retry_items = models.ImportItem.objects.filter(job=retry).all()
self.assertEqual(len(retry_items), 2)
self.assertEqual(retry_items[0].index, 0)
- self.assertEqual(import_items[0].data['Book Id'], '5498194')
+ self.assertEqual(import_items[0].data["Book Id"], "5498194")
self.assertEqual(retry_items[1].index, 1)
- self.assertEqual(retry_items[1].data['Book Id'], '5015319')
-
+ self.assertEqual(retry_items[1].data["Book Id"], "5015319")
@responses.activate
def test_import_data(self):
- ''' resolve entry '''
- import_job = self.importer.create_job(
- self.user, self.csv, False, 'unlisted')
- book = models.Edition.objects.create(title='Test Book')
+ """ resolve entry """
+ import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
+ book = models.Edition.objects.create(title="Test Book")
with patch(
- 'bookwyrm.models.import_job.ImportItem.get_book_from_isbn'
- ) as resolve:
+ "bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
+ ) as resolve:
resolve.return_value = book
- with patch('bookwyrm.importer.handle_imported_book'):
+ with patch("bookwyrm.importer.handle_imported_book"):
importer.import_data(self.importer.service, import_job.id)
import_item = models.ImportItem.objects.get(job=import_job, index=0)
self.assertEqual(import_item.book.id, book.id)
-
def test_handle_imported_book(self):
- ''' librarything import added a book, this adds related connections '''
- shelf = self.user.shelf_set.filter(identifier='read').first()
+ """ librarything import added a book, this adds related connections """
+ shelf = self.user.shelf_set.filter(identifier="read").first()
self.assertIsNone(shelf.books.first())
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/librarything.tsv')
- csv_file = open(datafile, 'r', encoding=self.importer.encoding)
- for index, entry in enumerate(list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))):
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
+ csv_file = open(datafile, "r", encoding=self.importer.encoding)
+ for index, entry in enumerate(
+ list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
+ ):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
@@ -133,31 +129,32 @@ class LibrarythingImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 5)
self.assertEqual(readthrough.finish_date.day, 8)
-
def test_handle_imported_book_already_shelved(self):
- ''' librarything import added a book, this adds related connections '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- shelf = self.user.shelf_set.filter(identifier='to-read').first()
- models.ShelfBook.objects.create(
- shelf=shelf, user=self.user, book=self.book)
+ """ librarything import added a book, this adds related connections """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ shelf = self.user.shelf_set.filter(identifier="to-read").first()
+ models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/librarything.tsv')
- csv_file = open(datafile, 'r', encoding=self.importer.encoding)
- for index, entry in enumerate(list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))):
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
+ csv_file = open(datafile, "r", encoding=self.importer.encoding)
+ for index, entry in enumerate(
+ list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
+ ):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
- self.assertIsNone(
- self.user.shelf_set.get(identifier='read').books.first())
+ self.assertIsNone(self.user.shelf_set.get(identifier="read").books.first())
readthrough = models.ReadThrough.objects.get(user=self.user)
self.assertEqual(readthrough.book, self.book)
self.assertEqual(readthrough.start_date.year, 2007)
@@ -167,24 +164,28 @@ class LibrarythingImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 5)
self.assertEqual(readthrough.finish_date.day, 8)
-
def test_handle_import_twice(self):
- ''' re-importing books '''
- shelf = self.user.shelf_set.filter(identifier='read').first()
+ """ re-importing books """
+ shelf = self.user.shelf_set.filter(identifier="read").first()
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/librarything.tsv')
- csv_file = open(datafile, 'r', encoding=self.importer.encoding)
- for index, entry in enumerate(list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))):
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
+ csv_file = open(datafile, "r", encoding=self.importer.encoding)
+ for index, entry in enumerate(
+ list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
+ ):
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=index, data=entry, book=self.book)
+ job_id=import_job.id, index=index, data=entry, book=self.book
+ )
break
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'public')
+ self.importer.service, self.user, import_item, False, "public"
+ )
shelf.refresh_from_db()
self.assertEqual(shelf.books.first(), self.book)
@@ -199,42 +200,44 @@ class LibrarythingImport(TestCase):
self.assertEqual(readthrough.finish_date.month, 5)
self.assertEqual(readthrough.finish_date.day, 8)
-
def test_handle_imported_book_review(self):
- ''' librarything review import '''
+ """ librarything review import """
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/librarything.tsv')
- csv_file = open(datafile, 'r', encoding=self.importer.encoding)
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
+ csv_file = open(datafile, "r", encoding=self.importer.encoding)
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[0]
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=0, data=entry, book=self.book)
+ job_id=import_job.id, index=0, data=entry, book=self.book
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, True, 'unlisted')
+ self.importer.service, self.user, import_item, True, "unlisted"
+ )
review = models.Review.objects.get(book=self.book, user=self.user)
- self.assertEqual(review.content, 'chef d\'oeuvre')
+ self.assertEqual(review.content, "chef d'oeuvre")
self.assertEqual(review.rating, 5)
self.assertEqual(review.published_date.year, 2007)
self.assertEqual(review.published_date.month, 5)
self.assertEqual(review.published_date.day, 8)
- self.assertEqual(review.privacy, 'unlisted')
-
+ self.assertEqual(review.privacy, "unlisted")
def test_handle_imported_book_reviews_disabled(self):
- ''' librarything review import '''
+ """ librarything review import """
import_job = models.ImportJob.objects.create(user=self.user)
- datafile = pathlib.Path(__file__).parent.joinpath('data/librarything.tsv')
- csv_file = open(datafile, 'r', encoding=self.importer.encoding)
+ datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
+ csv_file = open(datafile, "r", encoding=self.importer.encoding)
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[2]
entry = self.importer.parse_fields(entry)
import_item = models.ImportItem.objects.create(
- job_id=import_job.id, index=0, data=entry, book=self.book)
+ job_id=import_job.id, index=0, data=entry, book=self.book
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
importer.handle_imported_book(
- self.importer.service, self.user, import_item, False, 'unlisted')
- self.assertFalse(models.Review.objects.filter(
- book=self.book, user=self.user
- ).exists())
+ self.importer.service, self.user, import_item, False, "unlisted"
+ )
+ self.assertFalse(
+ models.Review.objects.filter(book=self.book, user=self.user).exists()
+ )
diff --git a/bookwyrm/tests/test_sanitize_html.py b/bookwyrm/tests/test_sanitize_html.py
index 58d94311..2b3d0378 100644
--- a/bookwyrm/tests/test_sanitize_html.py
+++ b/bookwyrm/tests/test_sanitize_html.py
@@ -1,28 +1,30 @@
-''' make sure only valid html gets to the app '''
+""" make sure only valid html gets to the app """
from django.test import TestCase
from bookwyrm.sanitize_html import InputHtmlParser
+
class Sanitizer(TestCase):
- ''' sanitizer tests '''
+ """ sanitizer tests """
+
def test_no_html(self):
- ''' just text '''
- input_text = 'no html '
+ """ just text """
+ input_text = "no html "
parser = InputHtmlParser()
parser.feed(input_text)
output = parser.get_output()
self.assertEqual(input_text, output)
def test_valid_html(self):
- ''' leave the html untouched '''
- input_text = '
yes html '
+ """ leave the html untouched """
+ input_text = "
yes html "
parser = InputHtmlParser()
parser.feed(input_text)
output = parser.get_output()
self.assertEqual(input_text, output)
def test_valid_html_attrs(self):
- ''' and don't remove attributes '''
+ """ and don't remove attributes """
input_text = '
yes html '
parser = InputHtmlParser()
parser.feed(input_text)
@@ -30,23 +32,23 @@ class Sanitizer(TestCase):
self.assertEqual(input_text, output)
def test_invalid_html(self):
- ''' remove all html when the html is malformed '''
- input_text = '
yes html '
+ """ remove all html when the html is malformed """
+ input_text = "yes html "
parser = InputHtmlParser()
parser.feed(input_text)
output = parser.get_output()
- self.assertEqual('yes html', output)
+ self.assertEqual("yes html", output)
- input_text = 'yes html '
+ input_text = "yes html "
parser = InputHtmlParser()
parser.feed(input_text)
output = parser.get_output()
- self.assertEqual('yes html ', output)
+ self.assertEqual("yes html ", output)
def test_disallowed_html(self):
- ''' remove disallowed html but keep allowed html '''
- input_text = '
yes html
'
+ """ remove disallowed html but keep allowed html """
+ input_text = "
yes html
"
parser = InputHtmlParser()
parser.feed(input_text)
output = parser.get_output()
- self.assertEqual(' yes
html ', output)
+ self.assertEqual(" yes
html ", output)
diff --git a/bookwyrm/tests/test_signing.py b/bookwyrm/tests/test_signing.py
index f6de11e1..d9cc411c 100644
--- a/bookwyrm/tests/test_signing.py
+++ b/bookwyrm/tests/test_signing.py
@@ -1,4 +1,4 @@
-''' getting and verifying signatures '''
+""" getting and verifying signatures """
import time
from collections import namedtuple
from urllib.parse import urlsplit
@@ -18,141 +18,125 @@ from bookwyrm.activitypub import Follow
from bookwyrm.settings import DOMAIN
from bookwyrm.signatures import create_key_pair, make_signature, make_digest
+
def get_follow_activity(follower, followee):
- ''' generates a test activity '''
+ """ generates a test activity """
return Follow(
- id='https://test.com/user/follow/id',
+ id="https://test.com/user/follow/id",
actor=follower.remote_id,
object=followee.remote_id,
).serialize()
-KeyPair = namedtuple('KeyPair', ('private_key', 'public_key'))
-Sender = namedtuple('Sender', ('remote_id', 'key_pair'))
+
+KeyPair = namedtuple("KeyPair", ("private_key", "public_key"))
+Sender = namedtuple("Sender", ("remote_id", "key_pair"))
+
class Signature(TestCase):
- ''' signature test '''
+ """ signature test """
+
def setUp(self):
- ''' create users and test data '''
+ """ create users and test data """
self.mouse = models.User.objects.create_user(
- 'mouse@%s' % DOMAIN, 'mouse@example.com', '',
- local=True, localname='mouse')
+ "mouse@%s" % DOMAIN, "mouse@example.com", "", local=True, localname="mouse"
+ )
self.rat = models.User.objects.create_user(
- 'rat@%s' % DOMAIN, 'rat@example.com', '',
- local=True, localname='rat')
+ "rat@%s" % DOMAIN, "rat@example.com", "", local=True, localname="rat"
+ )
self.cat = models.User.objects.create_user(
- 'cat@%s' % DOMAIN, 'cat@example.com', '',
- local=True, localname='cat')
+ "cat@%s" % DOMAIN, "cat@example.com", "", local=True, localname="cat"
+ )
private_key, public_key = create_key_pair()
self.fake_remote = Sender(
- 'http://localhost/user/remote',
- KeyPair(private_key, public_key)
+ "http://localhost/user/remote", KeyPair(private_key, public_key)
)
models.SiteSettings.objects.create()
def send(self, signature, now, data, digest):
- ''' test request '''
+ """ test request """
c = Client()
return c.post(
urlsplit(self.rat.inbox).path,
data=data,
- content_type='application/json',
+ content_type="application/json",
**{
- 'HTTP_DATE': now,
- 'HTTP_SIGNATURE': signature,
- 'HTTP_DIGEST': digest,
- 'HTTP_CONTENT_TYPE': 'application/activity+json; charset=utf-8',
- 'HTTP_HOST': DOMAIN,
+ "HTTP_DATE": now,
+ "HTTP_SIGNATURE": signature,
+ "HTTP_DIGEST": digest,
+ "HTTP_CONTENT_TYPE": "application/activity+json; charset=utf-8",
+ "HTTP_HOST": DOMAIN,
}
)
- def send_test_request(#pylint: disable=too-many-arguments
- self,
- sender,
- signer=None,
- send_data=None,
- digest=None,
- date=None):
- ''' sends a follow request to the "rat" user '''
+ def send_test_request( # pylint: disable=too-many-arguments
+ self, sender, signer=None, send_data=None, digest=None, date=None
+ ):
+ """ sends a follow request to the "rat" user """
now = date or http_date()
data = json.dumps(get_follow_activity(sender, self.rat))
digest = digest or make_digest(data)
- signature = make_signature(
- signer or sender, self.rat.inbox, now, digest)
- with patch('bookwyrm.views.inbox.activity_task.delay'):
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ signature = make_signature(signer or sender, self.rat.inbox, now, digest)
+ with patch("bookwyrm.views.inbox.activity_task.delay"):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
return self.send(signature, now, send_data or data, digest)
def test_correct_signature(self):
- ''' this one should just work '''
+ """ this one should just work """
response = self.send_test_request(sender=self.mouse)
self.assertEqual(response.status_code, 200)
def test_wrong_signature(self):
- ''' Messages must be signed by the right actor.
- (cat cannot sign messages on behalf of mouse) '''
+ """Messages must be signed by the right actor.
+ (cat cannot sign messages on behalf of mouse)"""
response = self.send_test_request(sender=self.mouse, signer=self.cat)
self.assertEqual(response.status_code, 401)
@responses.activate
def test_remote_signer(self):
- ''' signtures for remote users '''
- datafile = pathlib.Path(__file__).parent.joinpath('data/ap_user.json')
+ """ signtures for remote users """
+ datafile = pathlib.Path(__file__).parent.joinpath("data/ap_user.json")
data = json.loads(datafile.read_bytes())
- data['id'] = self.fake_remote.remote_id
- data['publicKey']['publicKeyPem'] = self.fake_remote.key_pair.public_key
- del data['icon'] # Avoid having to return an avatar.
+ data["id"] = self.fake_remote.remote_id
+ data["publicKey"]["publicKeyPem"] = self.fake_remote.key_pair.public_key
+ del data["icon"] # Avoid having to return an avatar.
+ responses.add(responses.GET, self.fake_remote.remote_id, json=data, status=200)
+ responses.add(
+ responses.GET, "https://localhost/.well-known/nodeinfo", status=404
+ )
responses.add(
responses.GET,
- self.fake_remote.remote_id,
- json=data,
- status=200)
- responses.add(
- responses.GET,
- 'https://localhost/.well-known/nodeinfo',
- status=404)
- responses.add(
- responses.GET,
- 'https://example.com/user/mouse/outbox?page=true',
- json={'orderedItems': []},
- status=200
+ "https://example.com/user/mouse/outbox?page=true",
+ json={"orderedItems": []},
+ status=200,
)
- with patch('bookwyrm.models.user.get_remote_reviews.delay'):
+ with patch("bookwyrm.models.user.get_remote_reviews.delay"):
response = self.send_test_request(sender=self.fake_remote)
self.assertEqual(response.status_code, 200)
@responses.activate
def test_key_needs_refresh(self):
- ''' an out of date key should be updated and the new key work '''
- datafile = pathlib.Path(__file__).parent.joinpath('data/ap_user.json')
+ """ an out of date key should be updated and the new key work """
+ datafile = pathlib.Path(__file__).parent.joinpath("data/ap_user.json")
data = json.loads(datafile.read_bytes())
- data['id'] = self.fake_remote.remote_id
- data['publicKey']['publicKeyPem'] = self.fake_remote.key_pair.public_key
- del data['icon'] # Avoid having to return an avatar.
+ data["id"] = self.fake_remote.remote_id
+ data["publicKey"]["publicKeyPem"] = self.fake_remote.key_pair.public_key
+ del data["icon"] # Avoid having to return an avatar.
+ responses.add(responses.GET, self.fake_remote.remote_id, json=data, status=200)
responses.add(
- responses.GET,
- self.fake_remote.remote_id,
- json=data,
- status=200)
- responses.add(
- responses.GET,
- 'https://localhost/.well-known/nodeinfo',
- status=404)
+ responses.GET, "https://localhost/.well-known/nodeinfo", status=404
+ )
# Second and subsequent fetches get a different key:
key_pair = KeyPair(*create_key_pair())
new_sender = Sender(self.fake_remote.remote_id, key_pair)
- data['publicKey']['publicKeyPem'] = key_pair.public_key
- responses.add(
- responses.GET,
- self.fake_remote.remote_id,
- json=data,
- status=200)
+ data["publicKey"]["publicKeyPem"] = key_pair.public_key
+ responses.add(responses.GET, self.fake_remote.remote_id, json=data, status=200)
- with patch('bookwyrm.models.user.get_remote_reviews.delay'):
+ with patch("bookwyrm.models.user.get_remote_reviews.delay"):
# Key correct:
response = self.send_test_request(sender=self.fake_remote)
self.assertEqual(response.status_code, 200)
@@ -169,43 +153,42 @@ class Signature(TestCase):
response = self.send_test_request(sender=self.fake_remote)
self.assertEqual(response.status_code, 401)
-
@responses.activate
def test_nonexistent_signer(self):
- ''' fail when unable to look up signer '''
+ """ fail when unable to look up signer """
responses.add(
responses.GET,
self.fake_remote.remote_id,
- json={'error': 'not found'},
- status=404)
+ json={"error": "not found"},
+ status=404,
+ )
response = self.send_test_request(sender=self.fake_remote)
self.assertEqual(response.status_code, 401)
@pytest.mark.integration
def test_changed_data(self):
- '''Message data must match the digest header.'''
- with patch('bookwyrm.activitypub.resolve_remote_id'):
+ """Message data must match the digest header."""
+ with patch("bookwyrm.activitypub.resolve_remote_id"):
response = self.send_test_request(
- self.mouse,
- send_data=get_follow_activity(self.mouse, self.cat))
+ self.mouse, send_data=get_follow_activity(self.mouse, self.cat)
+ )
self.assertEqual(response.status_code, 401)
@pytest.mark.integration
def test_invalid_digest(self):
- ''' signature digest must be valid '''
- with patch('bookwyrm.activitypub.resolve_remote_id'):
+ """ signature digest must be valid """
+ with patch("bookwyrm.activitypub.resolve_remote_id"):
response = self.send_test_request(
- self.mouse,
- digest='SHA-256=AAAAAAAAAAAAAAAAAA')
+ self.mouse, digest="SHA-256=AAAAAAAAAAAAAAAAAA"
+ )
self.assertEqual(response.status_code, 401)
@pytest.mark.integration
def test_old_message(self):
- '''Old messages should be rejected to prevent replay attacks.'''
- with patch('bookwyrm.activitypub.resolve_remote_id'):
+ """Old messages should be rejected to prevent replay attacks."""
+ with patch("bookwyrm.activitypub.resolve_remote_id"):
response = self.send_test_request(
- self.mouse,
- date=http_date(time.time() - 301)
+ self.mouse, date=http_date(time.time() - 301)
)
self.assertEqual(response.status_code, 401)
diff --git a/bookwyrm/tests/test_templatetags.py b/bookwyrm/tests/test_templatetags.py
index 45c99344..6bdbd04c 100644
--- a/bookwyrm/tests/test_templatetags.py
+++ b/bookwyrm/tests/test_templatetags.py
@@ -1,4 +1,4 @@
-''' style fixes and lookups for templates '''
+""" style fixes and lookups for templates """
import re
from unittest.mock import patch
@@ -11,82 +11,85 @@ from bookwyrm.templatetags import bookwyrm_tags
class TemplateTags(TestCase):
- ''' lotta different things here '''
- def setUp(self):
- ''' create some filler objects '''
- self.user = models.User.objects.create_user(
- 'mouse@example.com', 'mouse@mouse.mouse', 'mouseword',
- local=True, localname='mouse')
- with patch('bookwyrm.models.user.set_remote_server.delay'):
- self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.rat', 'ratword',
- remote_id='http://example.com/rat', local=False)
- self.book = models.Edition.objects.create(title='Test Book')
+ """ lotta different things here """
+ def setUp(self):
+ """ create some filler objects """
+ self.user = models.User.objects.create_user(
+ "mouse@example.com",
+ "mouse@mouse.mouse",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
+ self.remote_user = models.User.objects.create_user(
+ "rat",
+ "rat@rat.rat",
+ "ratword",
+ remote_id="http://example.com/rat",
+ local=False,
+ )
+ self.book = models.Edition.objects.create(title="Test Book")
def test_dict_key(self):
- ''' just getting a value out of a dict '''
- test_dict = {'a': 1, 'b': 3}
- self.assertEqual(
- bookwyrm_tags.dict_key(test_dict, 'a'), 1)
- self.assertEqual(
- bookwyrm_tags.dict_key(test_dict, 'c'), 0)
-
+ """ just getting a value out of a dict """
+ test_dict = {"a": 1, "b": 3}
+ self.assertEqual(bookwyrm_tags.dict_key(test_dict, "a"), 1)
+ self.assertEqual(bookwyrm_tags.dict_key(test_dict, "c"), 0)
def test_get_user_rating(self):
- ''' get a user's most recent rating of a book '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- models.Review.objects.create(
- user=self.user, book=self.book, rating=3)
- self.assertEqual(
- bookwyrm_tags.get_user_rating(self.book, self.user), 3)
-
+ """ get a user's most recent rating of a book """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ models.Review.objects.create(user=self.user, book=self.book, rating=3)
+ self.assertEqual(bookwyrm_tags.get_user_rating(self.book, self.user), 3)
def test_get_user_rating_doesnt_exist(self):
- ''' there is no rating available '''
- self.assertEqual(
- bookwyrm_tags.get_user_rating(self.book, self.user), 0)
-
+ """ there is no rating available """
+ self.assertEqual(bookwyrm_tags.get_user_rating(self.book, self.user), 0)
def test_get_user_identifer_local(self):
- ''' fall back to the simplest uid available '''
+ """ fall back to the simplest uid available """
self.assertNotEqual(self.user.username, self.user.localname)
- self.assertEqual(
- bookwyrm_tags.get_user_identifier(self.user), 'mouse')
+ self.assertEqual(bookwyrm_tags.get_user_identifier(self.user), "mouse")
def test_get_user_identifer_remote(self):
- ''' for a remote user, should be their full username '''
+ """ for a remote user, should be their full username """
self.assertEqual(
- bookwyrm_tags.get_user_identifier(self.remote_user),
- 'rat@example.com')
+ bookwyrm_tags.get_user_identifier(self.remote_user), "rat@example.com"
+ )
def test_get_notification_count(self):
- ''' just countin' '''
+ """ just countin' """
self.assertEqual(bookwyrm_tags.get_notification_count(self.user), 0)
- models.Notification.objects.create(
- user=self.user, notification_type='FAVORITE')
- models.Notification.objects.create(
- user=self.user, notification_type='MENTION')
+ models.Notification.objects.create(user=self.user, notification_type="FAVORITE")
+ models.Notification.objects.create(user=self.user, notification_type="MENTION")
models.Notification.objects.create(
- user=self.remote_user, notification_type='FOLLOW')
+ user=self.remote_user, notification_type="FOLLOW"
+ )
self.assertEqual(bookwyrm_tags.get_notification_count(self.user), 2)
-
def test_get_replies(self):
- ''' direct replies to a status '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ direct replies to a status """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
parent = models.Review.objects.create(
- user=self.user, book=self.book, content='hi')
+ user=self.user, book=self.book, content="hi"
+ )
first_child = models.Status.objects.create(
- reply_parent=parent, user=self.user, content='hi')
+ reply_parent=parent, user=self.user, content="hi"
+ )
second_child = models.Status.objects.create(
- reply_parent=parent, user=self.user, content='hi')
+ reply_parent=parent, user=self.user, content="hi"
+ )
third_child = models.Status.objects.create(
- reply_parent=parent, user=self.user,
- deleted=True, deleted_date=timezone.now())
+ reply_parent=parent,
+ user=self.user,
+ deleted=True,
+ deleted_date=timezone.now(),
+ )
replies = bookwyrm_tags.get_replies(parent)
self.assertEqual(len(replies), 2)
@@ -94,181 +97,162 @@ class TemplateTags(TestCase):
self.assertTrue(second_child in replies)
self.assertFalse(third_child in replies)
-
def test_get_parent(self):
- ''' get the reply parent of a status '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ get the reply parent of a status """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
parent = models.Review.objects.create(
- user=self.user, book=self.book, content='hi')
+ user=self.user, book=self.book, content="hi"
+ )
child = models.Status.objects.create(
- reply_parent=parent, user=self.user, content='hi')
+ reply_parent=parent, user=self.user, content="hi"
+ )
result = bookwyrm_tags.get_parent(child)
self.assertEqual(result, parent)
self.assertIsInstance(result, models.Review)
-
def test_get_user_liked(self):
- ''' did a user like a status '''
- status = models.Review.objects.create(
- user=self.remote_user, book=self.book)
+ """ did a user like a status """
+ status = models.Review.objects.create(user=self.remote_user, book=self.book)
self.assertFalse(bookwyrm_tags.get_user_liked(self.user, status))
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- models.Favorite.objects.create(
- user=self.user,
- status=status
- )
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ models.Favorite.objects.create(user=self.user, status=status)
self.assertTrue(bookwyrm_tags.get_user_liked(self.user, status))
-
def test_get_user_boosted(self):
- ''' did a user boost a status '''
- status = models.Review.objects.create(
- user=self.remote_user, book=self.book)
+ """ did a user boost a status """
+ status = models.Review.objects.create(user=self.remote_user, book=self.book)
self.assertFalse(bookwyrm_tags.get_user_boosted(self.user, status))
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- models.Boost.objects.create(
- user=self.user,
- boosted_status=status
- )
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ models.Boost.objects.create(user=self.user, boosted_status=status)
self.assertTrue(bookwyrm_tags.get_user_boosted(self.user, status))
-
def test_follow_request_exists(self):
- ''' does a user want to follow '''
+ """ does a user want to follow """
self.assertFalse(
- bookwyrm_tags.follow_request_exists(self.user, self.remote_user))
+ bookwyrm_tags.follow_request_exists(self.user, self.remote_user)
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.UserFollowRequest.objects.create(
- user_subject=self.user,
- user_object=self.remote_user)
+ user_subject=self.user, user_object=self.remote_user
+ )
self.assertFalse(
- bookwyrm_tags.follow_request_exists(self.user, self.remote_user))
+ bookwyrm_tags.follow_request_exists(self.user, self.remote_user)
+ )
self.assertTrue(
- bookwyrm_tags.follow_request_exists(self.remote_user, self.user))
-
+ bookwyrm_tags.follow_request_exists(self.remote_user, self.user)
+ )
def test_get_boosted(self):
- ''' load a boosted status '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Review.objects.create(
- user=self.remote_user, book=self.book)
- boost = models.Boost.objects.create(
- user=self.user,
- boosted_status=status
- )
+ """ load a boosted status """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Review.objects.create(user=self.remote_user, book=self.book)
+ boost = models.Boost.objects.create(user=self.user, boosted_status=status)
boosted = bookwyrm_tags.get_boosted(boost)
self.assertIsInstance(boosted, models.Review)
self.assertEqual(boosted, status)
-
def test_get_book_description(self):
- ''' grab it from the edition or the parent '''
- work = models.Work.objects.create(title='Test Work')
+ """ grab it from the edition or the parent """
+ work = models.Work.objects.create(title="Test Work")
self.book.parent_work = work
self.book.save()
self.assertIsNone(bookwyrm_tags.get_book_description(self.book))
- work.description = 'hi'
+ work.description = "hi"
work.save()
- self.assertEqual(bookwyrm_tags.get_book_description(self.book), 'hi')
+ self.assertEqual(bookwyrm_tags.get_book_description(self.book), "hi")
- self.book.description = 'hello'
+ self.book.description = "hello"
self.book.save()
- self.assertEqual(bookwyrm_tags.get_book_description(self.book), 'hello')
-
+ self.assertEqual(bookwyrm_tags.get_book_description(self.book), "hello")
def test_get_uuid(self):
- ''' uuid functionality '''
- uuid = bookwyrm_tags.get_uuid('hi')
- self.assertTrue(re.match(r'hi[A-Za-z0-9\-]', uuid))
-
+ """ uuid functionality """
+ uuid = bookwyrm_tags.get_uuid("hi")
+ self.assertTrue(re.match(r"hi[A-Za-z0-9\-]", uuid))
def test_time_since(self):
- ''' ultraconcise timestamps '''
- self.assertEqual(bookwyrm_tags.time_since('bleh'), '')
+ """ ultraconcise timestamps """
+ self.assertEqual(bookwyrm_tags.time_since("bleh"), "")
now = timezone.now()
- self.assertEqual(bookwyrm_tags.time_since(now), '0s')
+ self.assertEqual(bookwyrm_tags.time_since(now), "0s")
seconds_ago = now - relativedelta(seconds=4)
- self.assertEqual(bookwyrm_tags.time_since(seconds_ago), '4s')
+ self.assertEqual(bookwyrm_tags.time_since(seconds_ago), "4s")
minutes_ago = now - relativedelta(minutes=8)
- self.assertEqual(bookwyrm_tags.time_since(minutes_ago), '8m')
+ self.assertEqual(bookwyrm_tags.time_since(minutes_ago), "8m")
hours_ago = now - relativedelta(hours=9)
- self.assertEqual(bookwyrm_tags.time_since(hours_ago), '9h')
+ self.assertEqual(bookwyrm_tags.time_since(hours_ago), "9h")
days_ago = now - relativedelta(days=3)
- self.assertEqual(bookwyrm_tags.time_since(days_ago), '3d')
+ self.assertEqual(bookwyrm_tags.time_since(days_ago), "3d")
# I am not going to figure out how to mock dates tonight.
months_ago = now - relativedelta(months=5)
- self.assertTrue(re.match(
- r'[A-Z][a-z]{2} \d?\d',
- bookwyrm_tags.time_since(months_ago)
- ))
+ self.assertTrue(
+ re.match(r"[A-Z][a-z]{2} \d?\d", bookwyrm_tags.time_since(months_ago))
+ )
years_ago = now - relativedelta(years=10)
- self.assertTrue(re.match(
- r'[A-Z][a-z]{2} \d?\d \d{4}',
- bookwyrm_tags.time_since(years_ago)
- ))
-
+ self.assertTrue(
+ re.match(r"[A-Z][a-z]{2} \d?\d \d{4}", bookwyrm_tags.time_since(years_ago))
+ )
def test_get_markdown(self):
- ''' mardown format data '''
- result = bookwyrm_tags.get_markdown('_hi_')
- self.assertEqual(result, '
hi
')
-
- result = bookwyrm_tags.get_markdown('
_hi_ ')
- self.assertEqual(result, '
hi
')
+ """ mardown format data """
+ result = bookwyrm_tags.get_markdown("_hi_")
+ self.assertEqual(result, "
hi
")
+ result = bookwyrm_tags.get_markdown("
_hi_ ")
+ self.assertEqual(result, "
hi
")
def test_get_mentions(self):
- ''' list of people mentioned '''
- status = models.Status.objects.create(
- content='hi', user=self.remote_user)
+ """ list of people mentioned """
+ status = models.Status.objects.create(content="hi", user=self.remote_user)
result = bookwyrm_tags.get_mentions(status, self.user)
- self.assertEqual(result, '@rat@example.com ')
-
+ self.assertEqual(result, "@rat@example.com ")
def test_get_status_preview_name(self):
- ''' status context string '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(content='hi', user=self.user)
+ """ status context string """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(content="hi", user=self.user)
result = bookwyrm_tags.get_status_preview_name(status)
- self.assertEqual(result, 'status')
+ self.assertEqual(result, "status")
status = models.Review.objects.create(
- content='hi', user=self.user, book=self.book)
+ content="hi", user=self.user, book=self.book
+ )
result = bookwyrm_tags.get_status_preview_name(status)
- self.assertEqual(result, 'review of
Test Book ')
+ self.assertEqual(result, "review of
Test Book ")
status = models.Comment.objects.create(
- content='hi', user=self.user, book=self.book)
+ content="hi", user=self.user, book=self.book
+ )
result = bookwyrm_tags.get_status_preview_name(status)
- self.assertEqual(result, 'comment on
Test Book ')
+ self.assertEqual(result, "comment on
Test Book ")
status = models.Quotation.objects.create(
- content='hi', user=self.user, book=self.book)
+ content="hi", user=self.user, book=self.book
+ )
result = bookwyrm_tags.get_status_preview_name(status)
- self.assertEqual(result, 'quotation from
Test Book ')
-
+ self.assertEqual(result, "quotation from
Test Book ")
def test_related_status(self):
- ''' gets the subclass model for a notification status '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(content='hi', user=self.user)
+ """ gets the subclass model for a notification status """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(content="hi", user=self.user)
notification = models.Notification.objects.create(
- user=self.user, notification_type='MENTION',
- related_status=status)
+ user=self.user, notification_type="MENTION", related_status=status
+ )
result = bookwyrm_tags.related_status(notification)
self.assertIsInstance(result, models.Status)
diff --git a/bookwyrm/tests/views/test_authentication.py b/bookwyrm/tests/views/test_authentication.py
index dc52719c..f6d31861 100644
--- a/bookwyrm/tests/views/test_authentication.py
+++ b/bookwyrm/tests/views/test_authentication.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import AnonymousUser
@@ -14,21 +14,26 @@ from bookwyrm.settings import DOMAIN
# pylint: disable=too-many-public-methods
class AuthenticationViews(TestCase):
- ''' login and password management '''
+ """ login and password management """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.com', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "password",
+ local=True,
+ localname="mouse",
+ )
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
self.settings = models.SiteSettings.objects.create(id=1)
def test_login_get(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
login = views.Login.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.anonymous_user
result = login(request)
@@ -38,135 +43,117 @@ class AuthenticationViews(TestCase):
request.user = self.local_user
result = login(request)
- self.assertEqual(result.url, '/')
+ self.assertEqual(result.url, "/")
self.assertEqual(result.status_code, 302)
-
def test_register(self):
- ''' create a user '''
+ """ create a user """
view = views.Register.as_view()
self.assertEqual(models.User.objects.count(), 1)
request = self.factory.post(
- 'register/',
+ "register/",
{
- 'localname': 'nutria-user.user_nutria',
- 'password': 'mouseword',
- 'email': 'aa@bb.cccc'
- })
- with patch('bookwyrm.views.authentication.login'):
+ "localname": "nutria-user.user_nutria",
+ "password": "mouseword",
+ "email": "aa@bb.cccc",
+ },
+ )
+ with patch("bookwyrm.views.authentication.login"):
response = view(request)
self.assertEqual(models.User.objects.count(), 2)
self.assertEqual(response.status_code, 302)
nutria = models.User.objects.last()
- self.assertEqual(nutria.username, 'nutria-user.user_nutria@%s' % DOMAIN)
- self.assertEqual(nutria.localname, 'nutria-user.user_nutria')
+ self.assertEqual(nutria.username, "nutria-user.user_nutria@%s" % DOMAIN)
+ self.assertEqual(nutria.localname, "nutria-user.user_nutria")
self.assertEqual(nutria.local, True)
def test_register_trailing_space(self):
- ''' django handles this so weirdly '''
+ """ django handles this so weirdly """
view = views.Register.as_view()
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nutria ',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc'
- })
- with patch('bookwyrm.views.authentication.login'):
+ "register/",
+ {"localname": "nutria ", "password": "mouseword", "email": "aa@bb.ccc"},
+ )
+ with patch("bookwyrm.views.authentication.login"):
response = view(request)
self.assertEqual(models.User.objects.count(), 2)
self.assertEqual(response.status_code, 302)
nutria = models.User.objects.last()
- self.assertEqual(nutria.username, 'nutria@%s' % DOMAIN)
- self.assertEqual(nutria.localname, 'nutria')
+ self.assertEqual(nutria.username, "nutria@%s" % DOMAIN)
+ self.assertEqual(nutria.localname, "nutria")
self.assertEqual(nutria.local, True)
def test_register_invalid_email(self):
- ''' gotta have an email '''
+ """ gotta have an email """
view = views.Register.as_view()
self.assertEqual(models.User.objects.count(), 1)
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nutria',
- 'password': 'mouseword',
- 'email': 'aa'
- })
+ "register/", {"localname": "nutria", "password": "mouseword", "email": "aa"}
+ )
response = view(request)
self.assertEqual(models.User.objects.count(), 1)
response.render()
def test_register_invalid_username(self):
- ''' gotta have an email '''
+ """ gotta have an email """
view = views.Register.as_view()
self.assertEqual(models.User.objects.count(), 1)
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nut@ria',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc'
- })
+ "register/",
+ {"localname": "nut@ria", "password": "mouseword", "email": "aa@bb.ccc"},
+ )
response = view(request)
self.assertEqual(models.User.objects.count(), 1)
response.render()
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nutr ia',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc'
- })
+ "register/",
+ {"localname": "nutr ia", "password": "mouseword", "email": "aa@bb.ccc"},
+ )
response = view(request)
self.assertEqual(models.User.objects.count(), 1)
response.render()
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nut@ria',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc'
- })
+ "register/",
+ {"localname": "nut@ria", "password": "mouseword", "email": "aa@bb.ccc"},
+ )
response = view(request)
self.assertEqual(models.User.objects.count(), 1)
response.render()
-
def test_register_closed_instance(self):
- ''' you can't just register '''
+ """ you can't just register """
view = views.Register.as_view()
self.settings.allow_registration = False
self.settings.save()
request = self.factory.post(
- 'register/',
- {
- 'localname': 'nutria ',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc'
- })
+ "register/",
+ {"localname": "nutria ", "password": "mouseword", "email": "aa@bb.ccc"},
+ )
with self.assertRaises(PermissionDenied):
view(request)
def test_register_invite(self):
- ''' you can't just register '''
+ """ you can't just register """
view = views.Register.as_view()
self.settings.allow_registration = False
self.settings.save()
models.SiteInvite.objects.create(
- code='testcode', user=self.local_user, use_limit=1)
+ code="testcode", user=self.local_user, use_limit=1
+ )
self.assertEqual(models.SiteInvite.objects.get().times_used, 0)
request = self.factory.post(
- 'register/',
+ "register/",
{
- 'localname': 'nutria',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc',
- 'invite_code': 'testcode'
- })
- with patch('bookwyrm.views.authentication.login'):
+ "localname": "nutria",
+ "password": "mouseword",
+ "email": "aa@bb.ccc",
+ "invite_code": "testcode",
+ },
+ )
+ with patch("bookwyrm.views.authentication.login"):
response = view(request)
self.assertEqual(models.User.objects.count(), 2)
self.assertEqual(response.status_code, 302)
@@ -174,26 +161,28 @@ class AuthenticationViews(TestCase):
# invite already used to max capacity
request = self.factory.post(
- 'register/',
+ "register/",
{
- 'localname': 'nutria2',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc',
- 'invite_code': 'testcode'
- })
+ "localname": "nutria2",
+ "password": "mouseword",
+ "email": "aa@bb.ccc",
+ "invite_code": "testcode",
+ },
+ )
with self.assertRaises(PermissionDenied):
response = view(request)
self.assertEqual(models.User.objects.count(), 2)
# bad invite code
request = self.factory.post(
- 'register/',
+ "register/",
{
- 'localname': 'nutria3',
- 'password': 'mouseword',
- 'email': 'aa@bb.ccc',
- 'invite_code': 'dkfkdjgdfkjgkdfj'
- })
+ "localname": "nutria3",
+ "password": "mouseword",
+ "email": "aa@bb.ccc",
+ "invite_code": "dkfkdjgdfkjgkdfj",
+ },
+ )
with self.assertRaises(Http404):
response = view(request)
self.assertEqual(models.User.objects.count(), 2)
diff --git a/bookwyrm/tests/views/test_author.py b/bookwyrm/tests/views/test_author.py
index 3c1a68bb..bb047b7c 100644
--- a/bookwyrm/tests/views/test_author.py
+++ b/bookwyrm/tests/views/test_author.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import Group, Permission
from django.contrib.contenttypes.models import ContentType
@@ -12,37 +12,41 @@ from bookwyrm.activitypub import ActivitypubResponse
class AuthorViews(TestCase):
- ''' author views'''
+ """ author views"""
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.group = Group.objects.create(name='editor')
+ self.group = Group.objects.create(name="editor")
self.group.permissions.add(
Permission.objects.create(
- name='edit_book',
- codename='edit_book',
- content_type=ContentType.objects.get_for_model(models.User)).id
+ name="edit_book",
+ codename="edit_book",
+ content_type=ContentType.objects.get_for_model(models.User),
+ ).id
)
- self.work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
models.SiteSettings.objects.create()
-
def test_author_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Author.as_view()
- author = models.Author.objects.create(name='Jessica')
- request = self.factory.get('')
- with patch('bookwyrm.views.author.is_api_request') as is_api:
+ author = models.Author.objects.create(name="Jessica")
+ request = self.factory.get("")
+ with patch("bookwyrm.views.author.is_api_request") as is_api:
is_api.return_value = False
result = view(request, author.id)
self.assertIsInstance(result, TemplateResponse)
@@ -50,19 +54,18 @@ class AuthorViews(TestCase):
self.assertEqual(result.status_code, 200)
self.assertEqual(result.status_code, 200)
- request = self.factory.get('')
- with patch('bookwyrm.views.author.is_api_request') as is_api:
+ request = self.factory.get("")
+ with patch("bookwyrm.views.author.is_api_request") as is_api:
is_api.return_value = True
result = view(request, author.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_edit_author_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.EditAuthor.as_view()
- author = models.Author.objects.create(name='Test Author')
- request = self.factory.get('')
+ author = models.Author.objects.create(name="Test Author")
+ request = self.factory.get("")
request.user = self.local_user
request.user.is_superuser = True
@@ -72,52 +75,51 @@ class AuthorViews(TestCase):
self.assertEqual(result.status_code, 200)
self.assertEqual(result.status_code, 200)
-
def test_edit_author(self):
- ''' edit an author '''
+ """ edit an author """
view = views.EditAuthor.as_view()
- author = models.Author.objects.create(name='Test Author')
+ author = models.Author.objects.create(name="Test Author")
self.local_user.groups.add(self.group)
form = forms.AuthorForm(instance=author)
- form.data['name'] = 'New Name'
- form.data['last_edited_by'] = self.local_user.id
- request = self.factory.post('', form.data)
+ form.data["name"] = "New Name"
+ form.data["last_edited_by"] = self.local_user.id
+ request = self.factory.post("", form.data)
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, author.id)
author.refresh_from_db()
- self.assertEqual(author.name, 'New Name')
+ self.assertEqual(author.name, "New Name")
self.assertEqual(author.last_edited_by, self.local_user)
def test_edit_author_non_editor(self):
- ''' edit an author with invalid post data'''
+ """ edit an author with invalid post data"""
view = views.EditAuthor.as_view()
- author = models.Author.objects.create(name='Test Author')
+ author = models.Author.objects.create(name="Test Author")
form = forms.AuthorForm(instance=author)
- form.data['name'] = 'New Name'
- form.data['last_edited_by'] = self.local_user.id
- request = self.factory.post('', form.data)
+ form.data["name"] = "New Name"
+ form.data["last_edited_by"] = self.local_user.id
+ request = self.factory.post("", form.data)
request.user = self.local_user
with self.assertRaises(PermissionDenied):
view(request, author.id)
author.refresh_from_db()
- self.assertEqual(author.name, 'Test Author')
+ self.assertEqual(author.name, "Test Author")
def test_edit_author_invalid_form(self):
- ''' edit an author with invalid post data'''
+ """ edit an author with invalid post data"""
view = views.EditAuthor.as_view()
- author = models.Author.objects.create(name='Test Author')
+ author = models.Author.objects.create(name="Test Author")
self.local_user.groups.add(self.group)
form = forms.AuthorForm(instance=author)
- form.data['name'] = ''
- form.data['last_edited_by'] = self.local_user.id
- request = self.factory.post('', form.data)
+ form.data["name"] = ""
+ form.data["last_edited_by"] = self.local_user.id
+ request = self.factory.post("", form.data)
request.user = self.local_user
resp = view(request, author.id)
author.refresh_from_db()
- self.assertEqual(author.name, 'Test Author')
+ self.assertEqual(author.name, "Test Author")
resp.render()
self.assertEqual(resp.status_code, 200)
diff --git a/bookwyrm/tests/views/test_block.py b/bookwyrm/tests/views/test_block.py
index 315dc224..60920e38 100644
--- a/bookwyrm/tests/views/test_block.py
+++ b/bookwyrm/tests/views/test_block.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.template.response import TemplateResponse
from django.test import TestCase
@@ -8,28 +8,34 @@ from bookwyrm import models, views
class BlockViews(TestCase):
- ''' view user and edit profile '''
+ """ view user and edit profile """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
models.SiteSettings.objects.create()
-
def test_block_get(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Block.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
@@ -37,19 +43,19 @@ class BlockViews(TestCase):
self.assertEqual(result.status_code, 200)
def test_block_post(self):
- ''' create a "block" database entry from an activity '''
+ """ create a "block" database entry from an activity """
view = views.Block.as_view()
self.local_user.followers.add(self.remote_user)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.UserFollowRequest.objects.create(
- user_subject=self.local_user,
- user_object=self.remote_user)
+ user_subject=self.local_user, user_object=self.remote_user
+ )
self.assertTrue(models.UserFollows.objects.exists())
self.assertTrue(models.UserFollowRequest.objects.exists())
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.remote_user.id)
block = models.UserBlocks.objects.get()
self.assertEqual(block.user_subject, self.local_user)
@@ -59,12 +65,12 @@ class BlockViews(TestCase):
self.assertFalse(models.UserFollowRequest.objects.exists())
def test_unblock(self):
- ''' undo a block '''
+ """ undo a block """
self.local_user.blocks.add(self.remote_user)
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.block.unblock(request, self.remote_user.id)
self.assertFalse(models.UserBlocks.objects.exists())
diff --git a/bookwyrm/tests/views/test_book.py b/bookwyrm/tests/views/test_book.py
index b3360200..eb8c89b9 100644
--- a/bookwyrm/tests/views/test_book.py
+++ b/bookwyrm/tests/views/test_book.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import Group, Permission
from django.contrib.contenttypes.models import ContentType
@@ -11,55 +11,58 @@ from bookwyrm.activitypub import ActivitypubResponse
class BookViews(TestCase):
- ''' books books books '''
+ """ books books books """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.group = Group.objects.create(name='editor')
+ self.group = Group.objects.create(name="editor")
self.group.permissions.add(
Permission.objects.create(
- name='edit_book',
- codename='edit_book',
- content_type=ContentType.objects.get_for_model(models.User)).id
+ name="edit_book",
+ codename="edit_book",
+ content_type=ContentType.objects.get_for_model(models.User),
+ ).id
)
- self.work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
models.SiteSettings.objects.create()
-
def test_book_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Book.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.books.is_api_request') as is_api:
+ with patch("bookwyrm.views.books.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.book.id)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- request = self.factory.get('')
- with patch('bookwyrm.views.books.is_api_request') as is_api:
+ request = self.factory.get("")
+ with patch("bookwyrm.views.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)
-
def test_edit_book_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.EditBook.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
request.user.is_superuser = True
result = view(request, self.book.id)
@@ -67,66 +70,57 @@ class BookViews(TestCase):
result.render()
self.assertEqual(result.status_code, 200)
-
def test_edit_book(self):
- ''' lets a user edit a book '''
+ """ lets a user edit a book """
view = views.EditBook.as_view()
self.local_user.groups.add(self.group)
form = forms.EditionForm(instance=self.book)
- form.data['title'] = 'New Title'
- form.data['last_edited_by'] = self.local_user.id
- request = self.factory.post('', form.data)
+ 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
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.book.id)
self.book.refresh_from_db()
- self.assertEqual(self.book.title, 'New Title')
-
+ self.assertEqual(self.book.title, "New Title")
def test_switch_edition(self):
- ''' updates user's relationships to a book '''
- work = models.Work.objects.create(title='test work')
- edition1 = models.Edition.objects.create(
- title='first ed', parent_work=work)
- edition2 = models.Edition.objects.create(
- title='second ed', parent_work=work)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- shelf = models.Shelf.objects.create(
- name='Test Shelf', user=self.local_user)
+ """ updates user's relationships to a book """
+ work = models.Work.objects.create(title="test work")
+ edition1 = models.Edition.objects.create(title="first ed", parent_work=work)
+ edition2 = models.Edition.objects.create(title="second ed", parent_work=work)
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ shelf = models.Shelf.objects.create(name="Test Shelf", user=self.local_user)
models.ShelfBook.objects.create(
book=edition1,
user=self.local_user,
shelf=shelf,
)
- models.ReadThrough.objects.create(
- user=self.local_user, book=edition1)
+ models.ReadThrough.objects.create(user=self.local_user, book=edition1)
self.assertEqual(models.ShelfBook.objects.get().book, edition1)
self.assertEqual(models.ReadThrough.objects.get().book, edition1)
- request = self.factory.post('', {
- 'edition': edition2.id
- })
+ request = self.factory.post("", {"edition": edition2.id})
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.switch_edition(request)
self.assertEqual(models.ShelfBook.objects.get().book, edition2)
self.assertEqual(models.ReadThrough.objects.get().book, edition2)
-
def test_editions_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Editions.as_view()
- request = self.factory.get('')
- with patch('bookwyrm.views.books.is_api_request') as is_api:
+ request = self.factory.get("")
+ with patch("bookwyrm.views.books.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.work.id)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- request = self.factory.get('')
- with patch('bookwyrm.views.books.is_api_request') as is_api:
+ request = self.factory.get("")
+ with patch("bookwyrm.views.books.is_api_request") as is_api:
is_api.return_value = True
result = view(request, self.work.id)
self.assertIsInstance(result, ActivitypubResponse)
diff --git a/bookwyrm/tests/views/test_federation.py b/bookwyrm/tests/views/test_federation.py
index 70cf41f6..d11d7415 100644
--- a/bookwyrm/tests/views/test_federation.py
+++ b/bookwyrm/tests/views/test_federation.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from django.template.response import TemplateResponse
from django.test import TestCase
from django.test.client import RequestFactory
@@ -8,20 +8,24 @@ from bookwyrm import views
class FederationViews(TestCase):
- ''' every response to a get request, html or json '''
+ """ every response to a get request, html or json """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
models.SiteSettings.objects.create()
-
def test_federation_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Federation.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
request.user.is_superuser = True
result = view(request)
diff --git a/bookwyrm/tests/views/test_feed.py b/bookwyrm/tests/views/test_feed.py
index 93af9944..c54be006 100644
--- a/bookwyrm/tests/views/test_feed.py
+++ b/bookwyrm/tests/views/test_feed.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.template.response import TemplateResponse
from django.test import TestCase
@@ -10,95 +10,93 @@ from bookwyrm.activitypub import ActivitypubResponse
class FeedMessageViews(TestCase):
- ''' dms '''
+ """ dms """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
self.book = models.Edition.objects.create(
- parent_work=models.Work.objects.create(title='hi'),
- title='Example Edition',
- remote_id='https://example.com/book/1',
+ parent_work=models.Work.objects.create(title="hi"),
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
)
models.SiteSettings.objects.create()
-
def test_feed(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Feed.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- result = view(request, 'local')
+ result = view(request, "local")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_status_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Status.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- content='hi', user=self.local_user)
- request = self.factory.get('')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(content="hi", user=self.local_user)
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.feed.is_api_request') as is_api:
+ with patch("bookwyrm.views.feed.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse', status.id)
+ result = view(request, "mouse", status.id)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.feed.is_api_request') as is_api:
+ with patch("bookwyrm.views.feed.is_api_request") as is_api:
is_api.return_value = True
- result = view(request, 'mouse', status.id)
+ result = view(request, "mouse", status.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_replies_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Replies.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- content='hi', user=self.local_user)
- request = self.factory.get('')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(content="hi", user=self.local_user)
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.feed.is_api_request') as is_api:
+ with patch("bookwyrm.views.feed.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse', status.id)
+ result = view(request, "mouse", status.id)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.feed.is_api_request') as is_api:
+ with patch("bookwyrm.views.feed.is_api_request") as is_api:
is_api.return_value = True
- result = view(request, 'mouse', status.id)
+ result = view(request, "mouse", status.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_direct_messages_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.DirectMessage.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_get_suggested_book(self):
- ''' gets books the ~*~ algorithm ~*~ thinks you want to post about '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ gets books the ~*~ algorithm ~*~ thinks you want to post about """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.ShelfBook.objects.create(
book=self.book,
user=self.local_user,
- shelf=self.local_user.shelf_set.get(identifier='reading')
+ shelf=self.local_user.shelf_set.get(identifier="reading"),
)
suggestions = views.feed.get_suggested_books(self.local_user)
- self.assertEqual(suggestions[0]['name'], 'Currently Reading')
- self.assertEqual(suggestions[0]['books'][0], self.book)
+ self.assertEqual(suggestions[0]["name"], "Currently Reading")
+ self.assertEqual(suggestions[0]["books"][0], self.book)
diff --git a/bookwyrm/tests/views/test_follow.py b/bookwyrm/tests/views/test_follow.py
index 62543d2d..67ac0f0b 100644
--- a/bookwyrm/tests/views/test_follow.py
+++ b/bookwyrm/tests/views/test_follow.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import Group, Permission
from django.contrib.contenttypes.models import ContentType
@@ -9,148 +9,147 @@ from bookwyrm import models, views
class BookViews(TestCase):
- ''' books books books '''
+ """ books books books """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- with patch('bookwyrm.models.user.set_remote_server'):
+ with patch("bookwyrm.models.user.set_remote_server"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@email.com', 'ratword',
+ "rat",
+ "rat@email.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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
- self.group = Group.objects.create(name='editor')
+ self.group = Group.objects.create(name="editor")
self.group.permissions.add(
Permission.objects.create(
- name='edit_book',
- codename='edit_book',
- content_type=ContentType.objects.get_for_model(models.User)).id
+ name="edit_book",
+ codename="edit_book",
+ content_type=ContentType.objects.get_for_model(models.User),
+ ).id
)
- self.work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
def test_handle_follow_remote(self):
- ''' send a follow request '''
- request = self.factory.post('', {'user': self.remote_user.username})
+ """ send a follow request """
+ request = self.factory.post("", {"user": self.remote_user.username})
request.user = self.local_user
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.follow(request)
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')
-
+ self.assertEqual(rel.status, "follow_request")
def test_handle_follow_local_manually_approves(self):
- ''' send a follow request '''
+ """ send a follow request """
rat = models.User.objects.create_user(
- 'rat@local.com', 'rat@rat.com', 'ratword',
- local=True, localname='rat',
- remote_id='https://example.com/users/rat',
+ "rat@local.com",
+ "rat@rat.com",
+ "ratword",
+ local=True,
+ localname="rat",
+ remote_id="https://example.com/users/rat",
manually_approves_followers=True,
)
- request = self.factory.post('', {'user': rat})
+ request = self.factory.post("", {"user": rat})
request.user = self.local_user
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.follow(request)
rel = models.UserFollowRequest.objects.get()
self.assertEqual(rel.user_subject, self.local_user)
self.assertEqual(rel.user_object, rat)
- self.assertEqual(rel.status, 'follow_request')
-
+ self.assertEqual(rel.status, "follow_request")
def test_handle_follow_local(self):
- ''' send a follow request '''
+ """ send a follow request """
rat = models.User.objects.create_user(
- 'rat@local.com', 'rat@rat.com', 'ratword',
- local=True, localname='rat',
- remote_id='https://example.com/users/rat',
+ "rat@local.com",
+ "rat@rat.com",
+ "ratword",
+ local=True,
+ localname="rat",
+ remote_id="https://example.com/users/rat",
)
- request = self.factory.post('', {'user': rat})
+ request = self.factory.post("", {"user": rat})
request.user = self.local_user
self.assertEqual(models.UserFollowRequest.objects.count(), 0)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.follow(request)
rel = models.UserFollows.objects.get()
self.assertEqual(rel.user_subject, self.local_user)
self.assertEqual(rel.user_object, rat)
- self.assertEqual(rel.status, 'follows')
-
+ self.assertEqual(rel.status, "follows")
def test_handle_unfollow(self):
- ''' send an unfollow '''
- request = self.factory.post('', {'user': self.remote_user.username})
+ """ send an unfollow """
+ request = self.factory.post("", {"user": self.remote_user.username})
request.user = self.local_user
self.remote_user.followers.add(self.local_user)
self.assertEqual(self.remote_user.followers.count(), 1)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
- as mock:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay") as mock:
views.unfollow(request)
self.assertEqual(mock.call_count, 1)
self.assertEqual(self.remote_user.followers.count(), 0)
-
def test_handle_accept(self):
- ''' accept a follow request '''
+ """ accept a follow request """
self.local_user.manually_approves_followers = True
self.local_user.save(broadcast=False)
- request = self.factory.post('', {'user': self.remote_user.username})
+ request = self.factory.post("", {"user": self.remote_user.username})
request.user = self.local_user
rel = models.UserFollowRequest.objects.create(
- user_subject=self.remote_user,
- user_object=self.local_user
+ user_subject=self.remote_user, user_object=self.local_user
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.accept_follow_request(request)
# request should be deleted
- self.assertEqual(
- models.UserFollowRequest.objects.filter(id=rel.id).count(), 0
- )
+ self.assertEqual(models.UserFollowRequest.objects.filter(id=rel.id).count(), 0)
# follow relationship should exist
self.assertEqual(self.local_user.followers.first(), self.remote_user)
-
def test_handle_reject(self):
- ''' reject a follow request '''
+ """ reject a follow request """
self.local_user.manually_approves_followers = True
self.local_user.save(broadcast=False)
- request = self.factory.post('', {'user': self.remote_user.username})
+ request = self.factory.post("", {"user": self.remote_user.username})
request.user = self.local_user
rel = models.UserFollowRequest.objects.create(
- user_subject=self.remote_user,
- user_object=self.local_user
+ user_subject=self.remote_user, user_object=self.local_user
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.delete_follow_request(request)
# request should be deleted
- self.assertEqual(
- models.UserFollowRequest.objects.filter(id=rel.id).count(), 0
- )
+ 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
- )
+ self.assertEqual(models.UserFollows.objects.filter(id=rel.id).count(), 0)
diff --git a/bookwyrm/tests/views/test_goal.py b/bookwyrm/tests/views/test_goal.py
index 0d534112..990bb5c2 100644
--- a/bookwyrm/tests/views/test_goal.py
+++ b/bookwyrm/tests/views/test_goal.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.utils import timezone
@@ -11,60 +11,65 @@ from bookwyrm import models, views
class GoalViews(TestCase):
- ''' viewing and creating statuses '''
+ """ viewing and creating statuses """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
self.rat = models.User.objects.create_user(
- 'rat@local.com', 'rat@rat.com', 'ratword',
- local=True, localname='rat',
- remote_id='https://example.com/users/rat',
+ "rat@local.com",
+ "rat@rat.com",
+ "ratword",
+ local=True,
+ localname="rat",
+ remote_id="https://example.com/users/rat",
)
self.book = models.Edition.objects.create(
- title='Example Edition',
- remote_id='https://example.com/book/1',
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
)
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
models.SiteSettings.objects.create()
-
def test_goal_page_no_goal(self):
- ''' view a reading goal page for another's unset goal '''
+ """ view a reading goal page for another's unset goal """
view = views.Goal.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.rat
result = view(request, self.local_user.localname, 2020)
self.assertEqual(result.status_code, 404)
def test_goal_page_no_goal_self(self):
- ''' view a reading goal page for your own unset goal '''
+ """ view a reading goal page for your own unset goal """
view = views.Goal.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request, self.local_user.localname, 2020)
result.render()
self.assertIsInstance(result, TemplateResponse)
-
def test_goal_page_anonymous(self):
- ''' can't view it without login '''
+ """ can't view it without login """
view = views.Goal.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.anonymous_user
result = view(request, self.local_user.localname, 2020)
self.assertEqual(result.status_code, 302)
def test_goal_page_public(self):
- ''' view a user's public goal '''
+ """ view a user's public goal """
models.ReadThrough.objects.create(
finish_date=timezone.now(),
user=self.local_user,
@@ -75,9 +80,10 @@ class GoalViews(TestCase):
user=self.local_user,
year=timezone.now().year,
goal=128937123,
- privacy='public')
+ privacy="public",
+ )
view = views.Goal.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.rat
result = view(request, self.local_user.localname, timezone.now().year)
@@ -85,40 +91,40 @@ class GoalViews(TestCase):
self.assertIsInstance(result, TemplateResponse)
def test_goal_page_private(self):
- ''' view a user's private goal '''
+ """ view a user's private goal """
models.AnnualGoal.objects.create(
- user=self.local_user,
- year=2020,
- goal=15,
- privacy='followers')
+ user=self.local_user, year=2020, goal=15, privacy="followers"
+ )
view = views.Goal.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.rat
result = view(request, self.local_user.localname, 2020)
self.assertEqual(result.status_code, 404)
-
def test_create_goal(self):
- ''' create a new goal '''
+ """ create a new goal """
view = views.Goal.as_view()
- request = self.factory.post('', {
- 'user': self.local_user.id,
- 'goal': 10,
- 'year': 2020,
- 'privacy': 'unlisted',
- 'post-status': True
- })
+ request = self.factory.post(
+ "",
+ {
+ "user": self.local_user.id,
+ "goal": 10,
+ "year": 2020,
+ "privacy": "unlisted",
+ "post-status": True,
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.local_user.localname, 2020)
goal = models.AnnualGoal.objects.get()
self.assertEqual(goal.user, self.local_user)
self.assertEqual(goal.goal, 10)
self.assertEqual(goal.year, 2020)
- self.assertEqual(goal.privacy, 'unlisted')
+ self.assertEqual(goal.privacy, "unlisted")
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
- self.assertEqual(status.privacy, 'unlisted')
+ self.assertEqual(status.privacy, "unlisted")
diff --git a/bookwyrm/tests/views/test_helpers.py b/bookwyrm/tests/views/test_helpers.py
index eff08307..bb4cf69c 100644
--- a/bookwyrm/tests/views/test_helpers.py
+++ b/bookwyrm/tests/views/test_helpers.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
import json
from unittest.mock import patch
import pathlib
@@ -9,127 +9,129 @@ import responses
from bookwyrm import models, views
from bookwyrm.settings import USER_AGENT
+
class ViewsHelpers(TestCase):
- ''' viewing and creating statuses '''
+ """ viewing and creating statuses """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.work = models.Work.objects.create(title='Test Work')
+ self.work = models.Work.objects.create(title="Test Work")
self.book = models.Edition.objects.create(
- title='Test Book',
- remote_id='https://example.com/book/1',
- parent_work=self.work
+ title="Test Book",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json'
- )
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
self.userdata = json.loads(datafile.read_bytes())
- del self.userdata['icon']
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ del self.userdata["icon"]
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.shelf = models.Shelf.objects.create(
- name='Test Shelf',
- identifier='test-shelf',
- user=self.local_user
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
)
-
def test_get_edition(self):
- ''' given an edition or a work, returns an edition '''
- self.assertEqual(
- views.helpers.get_edition(self.book.id), self.book)
- self.assertEqual(
- views.helpers.get_edition(self.work.id), self.book)
+ """ given an edition or a work, returns an edition """
+ self.assertEqual(views.helpers.get_edition(self.book.id), self.book)
+ self.assertEqual(views.helpers.get_edition(self.work.id), self.book)
def test_get_user_from_username(self):
- ''' works for either localname or username '''
+ """ works for either localname or username """
self.assertEqual(
- views.helpers.get_user_from_username(
- self.local_user, 'mouse'), self.local_user)
+ views.helpers.get_user_from_username(self.local_user, "mouse"),
+ self.local_user,
+ )
self.assertEqual(
- views.helpers.get_user_from_username(
- self.local_user, 'mouse@local.com'), self.local_user)
+ views.helpers.get_user_from_username(self.local_user, "mouse@local.com"),
+ self.local_user,
+ )
with self.assertRaises(models.User.DoesNotExist):
- views.helpers.get_user_from_username(
- self.local_user, 'mojfse@example.com')
-
+ views.helpers.get_user_from_username(self.local_user, "mojfse@example.com")
def test_is_api_request(self):
- ''' should it return html or json '''
- request = self.factory.get('/path')
- request.headers = {'Accept': 'application/json'}
+ """ should it return html or json """
+ request = self.factory.get("/path")
+ request.headers = {"Accept": "application/json"}
self.assertTrue(views.helpers.is_api_request(request))
- request = self.factory.get('/path.json')
- request.headers = {'Accept': 'Praise'}
+ request = self.factory.get("/path.json")
+ request.headers = {"Accept": "Praise"}
self.assertTrue(views.helpers.is_api_request(request))
- request = self.factory.get('/path')
- request.headers = {'Accept': 'Praise'}
+ request = self.factory.get("/path")
+ request.headers = {"Accept": "Praise"}
self.assertFalse(views.helpers.is_api_request(request))
-
def test_get_activity_feed(self):
- ''' loads statuses '''
+ """ loads statuses """
rat = models.User.objects.create_user(
- 'rat', 'rat@rat.rat', 'password', local=True)
+ "rat", "rat@rat.rat", "password", local=True
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
public_status = models.Comment.objects.create(
- content='public status', book=self.book, user=self.local_user)
+ content="public status", book=self.book, user=self.local_user
+ )
direct_status = models.Status.objects.create(
- content='direct', user=self.local_user, privacy='direct')
+ content="direct", user=self.local_user, privacy="direct"
+ )
- rat_public = models.Status.objects.create(
- content='blah blah', user=rat)
+ rat_public = models.Status.objects.create(content="blah blah", user=rat)
rat_unlisted = models.Status.objects.create(
- content='blah blah', user=rat, privacy='unlisted')
+ content="blah blah", user=rat, privacy="unlisted"
+ )
remote_status = models.Status.objects.create(
- content='blah blah', user=self.remote_user)
+ content="blah blah", user=self.remote_user
+ )
followers_status = models.Status.objects.create(
- content='blah', user=rat, privacy='followers')
+ content="blah", user=rat, privacy="followers"
+ )
rat_mention = models.Status.objects.create(
- content='blah blah blah', user=rat, privacy='followers')
+ content="blah blah blah", user=rat, privacy="followers"
+ )
rat_mention.mention_users.set([self.local_user])
statuses = views.helpers.get_activity_feed(
self.local_user,
- privacy=['public', 'unlisted', 'followers'],
+ privacy=["public", "unlisted", "followers"],
following_only=True,
- queryset=models.Comment.objects
+ queryset=models.Comment.objects,
)
self.assertEqual(len(statuses), 1)
self.assertEqual(statuses[0], public_status)
statuses = views.helpers.get_activity_feed(
- self.local_user,
- privacy=['public', 'followers'],
- local_only=True
+ self.local_user, privacy=["public", "followers"], local_only=True
)
self.assertEqual(len(statuses), 2)
self.assertEqual(statuses[1], public_status)
self.assertEqual(statuses[0], rat_public)
- statuses = views.helpers.get_activity_feed(
- self.local_user, privacy=['direct'])
+ statuses = views.helpers.get_activity_feed(self.local_user, privacy=["direct"])
self.assertEqual(len(statuses), 1)
self.assertEqual(statuses[0], direct_status)
statuses = views.helpers.get_activity_feed(
self.local_user,
- privacy=['public', 'followers'],
+ privacy=["public", "followers"],
)
self.assertEqual(len(statuses), 3)
self.assertEqual(statuses[2], public_status)
@@ -138,8 +140,8 @@ class ViewsHelpers(TestCase):
statuses = views.helpers.get_activity_feed(
self.local_user,
- privacy=['public', 'unlisted', 'followers'],
- following_only=True
+ privacy=["public", "unlisted", "followers"],
+ following_only=True,
)
self.assertEqual(len(statuses), 2)
self.assertEqual(statuses[1], public_status)
@@ -148,8 +150,8 @@ class ViewsHelpers(TestCase):
rat.followers.add(self.local_user)
statuses = views.helpers.get_activity_feed(
self.local_user,
- privacy=['public', 'unlisted', 'followers'],
- following_only=True
+ privacy=["public", "unlisted", "followers"],
+ following_only=True,
)
self.assertEqual(len(statuses), 5)
self.assertEqual(statuses[4], public_status)
@@ -158,187 +160,187 @@ class ViewsHelpers(TestCase):
self.assertEqual(statuses[1], followers_status)
self.assertEqual(statuses[0], rat_mention)
-
def test_get_activity_feed_blocks(self):
- ''' feed generation with blocked users '''
+ """ feed generation with blocked users """
rat = models.User.objects.create_user(
- 'rat', 'rat@rat.rat', 'password', local=True)
+ "rat", "rat@rat.rat", "password", local=True
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
public_status = models.Comment.objects.create(
- content='public status', book=self.book, user=self.local_user)
- rat_public = models.Status.objects.create(
- content='blah blah', user=rat)
+ content="public status", book=self.book, user=self.local_user
+ )
+ rat_public = models.Status.objects.create(content="blah blah", user=rat)
statuses = views.helpers.get_activity_feed(
- self.local_user, privacy=['public'])
+ self.local_user, privacy=["public"]
+ )
self.assertEqual(len(statuses), 2)
# block relationship
rat.blocks.add(self.local_user)
- statuses = views.helpers.get_activity_feed(
- self.local_user, privacy=['public'])
+ statuses = views.helpers.get_activity_feed(self.local_user, privacy=["public"])
self.assertEqual(len(statuses), 1)
self.assertEqual(statuses[0], public_status)
- statuses = views.helpers.get_activity_feed(
- rat, privacy=['public'])
+ statuses = views.helpers.get_activity_feed(rat, privacy=["public"])
self.assertEqual(len(statuses), 1)
self.assertEqual(statuses[0], rat_public)
-
-
def test_is_bookwyrm_request(self):
- ''' checks if a request came from a bookwyrm instance '''
- request = self.factory.get('', {'q': 'Test Book'})
+ """ checks if a request came from a bookwyrm instance """
+ request = self.factory.get("", {"q": "Test Book"})
self.assertFalse(views.helpers.is_bookwyrm_request(request))
request = self.factory.get(
- '', {'q': 'Test Book'},
- HTTP_USER_AGENT=\
- "http.rb/4.4.1 (Mastodon/3.3.0; +https://mastodon.social/)"
+ "",
+ {"q": "Test Book"},
+ HTTP_USER_AGENT="http.rb/4.4.1 (Mastodon/3.3.0; +https://mastodon.social/)",
)
self.assertFalse(views.helpers.is_bookwyrm_request(request))
- request = self.factory.get(
- '', {'q': 'Test Book'}, HTTP_USER_AGENT=USER_AGENT)
+ request = self.factory.get("", {"q": "Test Book"}, HTTP_USER_AGENT=USER_AGENT)
self.assertTrue(views.helpers.is_bookwyrm_request(request))
-
def test_existing_user(self):
- ''' simple database lookup by username '''
- result = views.helpers.handle_remote_webfinger('@mouse@local.com')
+ """ simple database lookup by username """
+ result = views.helpers.handle_remote_webfinger("@mouse@local.com")
self.assertEqual(result, self.local_user)
- result = views.helpers.handle_remote_webfinger('mouse@local.com')
+ result = views.helpers.handle_remote_webfinger("mouse@local.com")
self.assertEqual(result, self.local_user)
-
@responses.activate
def test_load_user(self):
- ''' find a remote user using webfinger '''
- username = 'mouse@example.com'
+ """ 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"
- }]
+ "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,
+ "https://example.com/.well-known/webfinger?resource=acct:%s" % username,
json=wellknown,
- status=200)
+ status=200,
+ )
responses.add(
responses.GET,
- 'https://example.com/user/mouse',
+ "https://example.com/user/mouse",
json=self.userdata,
- status=200)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
- result = views.helpers.handle_remote_webfinger('@mouse@example.com')
+ status=200,
+ )
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
+ result = views.helpers.handle_remote_webfinger("@mouse@example.com")
self.assertIsInstance(result, models.User)
- self.assertEqual(result.username, 'mouse@example.com')
-
+ self.assertEqual(result.username, "mouse@example.com")
def test_handle_reading_status_to_read(self):
- ''' posts shelve activities '''
- shelf = self.local_user.shelf_set.get(identifier='to-read')
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ posts shelve activities """
+ shelf = self.local_user.shelf_set.get(identifier="to-read")
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.helpers.handle_reading_status(
- self.local_user, shelf, self.book, 'public')
+ self.local_user, shelf, self.book, "public"
+ )
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.mention_books.first(), self.book)
- self.assertEqual(status.content, 'wants to read')
+ self.assertEqual(status.content, "wants to read")
def test_handle_reading_status_reading(self):
- ''' posts shelve activities '''
- shelf = self.local_user.shelf_set.get(identifier='reading')
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ posts shelve activities """
+ shelf = self.local_user.shelf_set.get(identifier="reading")
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.helpers.handle_reading_status(
- self.local_user, shelf, self.book, 'public')
+ self.local_user, shelf, self.book, "public"
+ )
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.mention_books.first(), self.book)
- self.assertEqual(status.content, 'started reading')
+ self.assertEqual(status.content, "started reading")
def test_handle_reading_status_read(self):
- ''' posts shelve activities '''
- shelf = self.local_user.shelf_set.get(identifier='read')
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ posts shelve activities """
+ shelf = self.local_user.shelf_set.get(identifier="read")
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.helpers.handle_reading_status(
- self.local_user, shelf, self.book, 'public')
+ self.local_user, shelf, self.book, "public"
+ )
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.mention_books.first(), self.book)
- self.assertEqual(status.content, 'finished reading')
+ self.assertEqual(status.content, "finished reading")
def test_handle_reading_status_other(self):
- ''' posts shelve activities '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ posts shelve activities """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.helpers.handle_reading_status(
- self.local_user, self.shelf, self.book, 'public')
+ self.local_user, self.shelf, self.book, "public"
+ )
self.assertFalse(models.GeneratedNote.objects.exists())
def test_object_visible_to_user(self):
- ''' does a user have permission to view an object '''
+ """ does a user have permission to view an object """
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='public')
- self.assertTrue(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="public"
+ )
+ self.assertTrue(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Shelf.objects.create(
- name='test', user=self.remote_user, privacy='unlisted')
- self.assertTrue(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ name="test", user=self.remote_user, privacy="unlisted"
+ )
+ self.assertTrue(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='followers')
- self.assertFalse(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="followers"
+ )
+ self.assertFalse(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='direct')
- self.assertFalse(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="direct"
+ )
+ self.assertFalse(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='direct')
+ content="hi", user=self.remote_user, privacy="direct"
+ )
obj.mention_users.add(self.local_user)
- self.assertTrue(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ self.assertTrue(views.helpers.object_visible_to_user(self.local_user, obj))
def test_object_visible_to_user_follower(self):
- ''' what you can see if you follow a user '''
+ """ what you can see if you follow a user """
self.remote_user.followers.add(self.local_user)
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='followers')
- self.assertTrue(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="followers"
+ )
+ self.assertTrue(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='direct')
- self.assertFalse(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="direct"
+ )
+ self.assertFalse(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='direct')
+ content="hi", user=self.remote_user, privacy="direct"
+ )
obj.mention_users.add(self.local_user)
- self.assertTrue(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ self.assertTrue(views.helpers.object_visible_to_user(self.local_user, obj))
def test_object_visible_to_user_blocked(self):
- ''' you can't see it if they block you '''
+ """ you can't see it if they block you """
self.remote_user.blocks.add(self.local_user)
obj = models.Status.objects.create(
- content='hi', user=self.remote_user, privacy='public')
- self.assertFalse(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ content="hi", user=self.remote_user, privacy="public"
+ )
+ self.assertFalse(views.helpers.object_visible_to_user(self.local_user, obj))
obj = models.Shelf.objects.create(
- name='test', user=self.remote_user, privacy='unlisted')
- self.assertFalse(
- views.helpers.object_visible_to_user(self.local_user, obj))
+ name="test", user=self.remote_user, privacy="unlisted"
+ )
+ self.assertFalse(views.helpers.object_visible_to_user(self.local_user, obj))
diff --git a/bookwyrm/tests/views/test_import.py b/bookwyrm/tests/views/test_import.py
index ba8f2457..b98b2516 100644
--- a/bookwyrm/tests/views/test_import.py
+++ b/bookwyrm/tests/views/test_import.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.template.response import TemplateResponse
from django.test import TestCase
@@ -9,34 +9,37 @@ from bookwyrm import views
class ImportViews(TestCase):
- ''' goodreads import views '''
+ """ goodreads import views """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
models.SiteSettings.objects.create()
-
def test_import_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Import.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_import_status(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.ImportStatus.as_view()
import_job = models.ImportJob.objects.create(user=self.local_user)
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.tasks.app.AsyncResult') as async_result:
+ with patch("bookwyrm.tasks.app.AsyncResult") as async_result:
async_result.return_value = []
result = view(request, import_job.id)
self.assertIsInstance(result, TemplateResponse)
diff --git a/bookwyrm/tests/views/test_inbox.py b/bookwyrm/tests/views/test_inbox.py
index ff55ad04..cab50892 100644
--- a/bookwyrm/tests/views/test_inbox.py
+++ b/bookwyrm/tests/views/test_inbox.py
@@ -1,4 +1,4 @@
-''' tests incoming activities'''
+""" tests incoming activities"""
from datetime import datetime
import json
import pathlib
@@ -11,101 +11,100 @@ import responses
from bookwyrm import models, views
-#pylint: disable=too-many-public-methods
+# pylint: disable=too-many-public-methods
class Inbox(TestCase):
- ''' readthrough tests '''
+ """ readthrough tests """
+
def setUp(self):
- ''' basic user and book data '''
+ """ basic user and book data """
self.client = Client()
self.local_user = models.User.objects.create_user(
- 'mouse@example.com', 'mouse@mouse.com', 'mouseword',
- local=True, localname='mouse')
- self.local_user.remote_id = 'https://example.com/user/mouse'
+ "mouse@example.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ )
+ self.local_user.remote_id = "https://example.com/user/mouse"
self.local_user.save(broadcast=False)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.status = models.Status.objects.create(
user=self.local_user,
- content='Test status',
- remote_id='https://example.com/status/1',
+ content="Test status",
+ remote_id="https://example.com/status/1",
)
self.create_json = {
- 'id': 'hi',
- 'type': 'Create',
- 'actor': 'hi',
- "to": [
- "https://www.w3.org/ns/activitystreams#public"
- ],
- "cc": [
- "https://example.com/user/mouse/followers"
- ],
- 'object': {}
+ "id": "hi",
+ "type": "Create",
+ "actor": "hi",
+ "to": ["https://www.w3.org/ns/activitystreams#public"],
+ "cc": ["https://example.com/user/mouse/followers"],
+ "object": {},
}
models.SiteSettings.objects.create()
-
def test_inbox_invalid_get(self):
- ''' shouldn't try to handle if the user is not found '''
- result = self.client.get(
- '/inbox', content_type="application/json"
- )
+ """ shouldn't try to handle if the user is not found """
+ result = self.client.get("/inbox", content_type="application/json")
self.assertIsInstance(result, HttpResponseNotAllowed)
def test_inbox_invalid_user(self):
- ''' shouldn't try to handle if the user is not found '''
+ """ shouldn't try to handle if the user is not found """
result = self.client.post(
- '/user/bleh/inbox',
+ "/user/bleh/inbox",
'{"type": "Test", "object": "exists"}',
- content_type="application/json"
+ content_type="application/json",
)
self.assertIsInstance(result, HttpResponseNotFound)
def test_inbox_invalid_bad_signature(self):
- ''' bad request for invalid signature '''
- with patch('bookwyrm.views.inbox.has_valid_signature') as mock_valid:
+ """ bad request for invalid signature """
+ with patch("bookwyrm.views.inbox.has_valid_signature") as mock_valid:
mock_valid.return_value = False
result = self.client.post(
- '/user/mouse/inbox',
- '{"type": "Test", "object": "exists"}',
- content_type="application/json"
+ "/user/mouse/inbox",
+ '{"type": "Announce", "object": "exists"}',
+ content_type="application/json",
)
self.assertEqual(result.status_code, 401)
def test_inbox_invalid_bad_signature_delete(self):
- ''' invalid signature for Delete is okay though '''
- with patch('bookwyrm.views.inbox.has_valid_signature') as mock_valid:
+ """ invalid signature for Delete is okay though """
+ with patch("bookwyrm.views.inbox.has_valid_signature") as mock_valid:
mock_valid.return_value = False
result = self.client.post(
- '/user/mouse/inbox',
+ "/user/mouse/inbox",
'{"type": "Delete", "object": "exists"}',
- content_type="application/json"
+ content_type="application/json",
)
self.assertEqual(result.status_code, 200)
def test_inbox_unknown_type(self):
- ''' never heard of that activity type, don't have a handler for it '''
- with patch('bookwyrm.views.inbox.has_valid_signature') as mock_valid:
+ """ never heard of that activity type, don't have a handler for it """
+ with patch("bookwyrm.views.inbox.has_valid_signature") as mock_valid:
result = self.client.post(
- '/inbox',
+ "/inbox",
'{"type": "Fish", "object": "exists"}',
- content_type="application/json"
+ content_type="application/json",
)
mock_valid.return_value = True
self.assertIsInstance(result, HttpResponseNotFound)
-
def test_inbox_success(self):
- ''' a known type, for which we start a task '''
+ """ a known type, for which we start a task """
activity = self.create_json
- activity['object'] = {
+ activity["object"] = {
"id": "https://example.com/list/22",
"type": "BookList",
"totalItems": 1,
@@ -113,47 +112,41 @@ class Inbox(TestCase):
"last": "https://example.com/list/22?page=1",
"name": "Test List",
"owner": "https://example.com/user/mouse",
- "to": [
- "https://www.w3.org/ns/activitystreams#Public"
- ],
- "cc": [
- "https://example.com/user/mouse/followers"
- ],
+ "to": ["https://www.w3.org/ns/activitystreams#Public"],
+ "cc": ["https://example.com/user/mouse/followers"],
"summary": "summary text",
"curation": "curated",
- "@context": "https://www.w3.org/ns/activitystreams"
+ "@context": "https://www.w3.org/ns/activitystreams",
}
- with patch('bookwyrm.views.inbox.has_valid_signature') as mock_valid:
+ with patch("bookwyrm.views.inbox.has_valid_signature") as mock_valid:
mock_valid.return_value = True
- with patch('bookwyrm.views.inbox.activity_task.delay'):
+ with patch("bookwyrm.views.inbox.activity_task.delay"):
result = self.client.post(
- '/inbox',
- json.dumps(activity),
- content_type="application/json"
+ "/inbox", json.dumps(activity), content_type="application/json"
)
self.assertEqual(result.status_code, 200)
-
def test_handle_create_status(self):
- ''' the "it justs works" mode '''
+ """ the "it justs works" mode """
self.assertEqual(models.Status.objects.count(), 1)
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_quotation.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_quotation.json")
status_data = json.loads(datafile.read_bytes())
models.Edition.objects.create(
- title='Test Book', remote_id='https://example.com/book/1')
+ title="Test Book", remote_id="https://example.com/book/1"
+ )
activity = self.create_json
- activity['object'] = status_data
+ activity["object"] = status_data
views.inbox.activity_task(activity)
status = models.Quotation.objects.get()
self.assertEqual(
- status.remote_id, 'https://example.com/user/mouse/quotation/13')
- self.assertEqual(status.quote, 'quote body')
- self.assertEqual(status.content, 'commentary')
+ status.remote_id, "https://example.com/user/mouse/quotation/13"
+ )
+ self.assertEqual(status.quote, "quote body")
+ self.assertEqual(status.content, "commentary")
self.assertEqual(status.user, self.local_user)
self.assertEqual(models.Status.objects.count(), 2)
@@ -161,56 +154,50 @@ class Inbox(TestCase):
views.inbox.activity_task(activity)
self.assertEqual(models.Status.objects.count(), 2)
-
def test_handle_create_status_remote_note_with_mention(self):
- ''' should only create it under the right circumstances '''
+ """ should only create it under the right circumstances """
self.assertEqual(models.Status.objects.count(), 1)
self.assertFalse(
- models.Notification.objects.filter(user=self.local_user).exists())
+ models.Notification.objects.filter(user=self.local_user).exists()
+ )
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_note.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_note.json")
status_data = json.loads(datafile.read_bytes())
activity = self.create_json
- activity['object'] = status_data
+ activity["object"] = status_data
views.inbox.activity_task(activity)
status = models.Status.objects.last()
- self.assertEqual(status.content, 'test content in note')
+ self.assertEqual(status.content, "test content in note")
self.assertEqual(status.mention_users.first(), self.local_user)
self.assertTrue(
- models.Notification.objects.filter(user=self.local_user).exists())
- self.assertEqual(
- models.Notification.objects.get().notification_type, 'MENTION')
+ models.Notification.objects.filter(user=self.local_user).exists()
+ )
+ self.assertEqual(models.Notification.objects.get().notification_type, "MENTION")
def test_handle_create_status_remote_note_with_reply(self):
- ''' should only create it under the right circumstances '''
+ """ should only create it under the right circumstances """
self.assertEqual(models.Status.objects.count(), 1)
- self.assertFalse(
- models.Notification.objects.filter(user=self.local_user))
+ self.assertFalse(models.Notification.objects.filter(user=self.local_user))
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_note.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_note.json")
status_data = json.loads(datafile.read_bytes())
- del status_data['tag']
- status_data['inReplyTo'] = self.status.remote_id
+ del status_data["tag"]
+ status_data["inReplyTo"] = self.status.remote_id
activity = self.create_json
- activity['object'] = status_data
+ activity["object"] = status_data
views.inbox.activity_task(activity)
status = models.Status.objects.last()
- self.assertEqual(status.content, 'test content in note')
+ self.assertEqual(status.content, "test content in note")
self.assertEqual(status.reply_parent, self.status)
- self.assertTrue(
- models.Notification.objects.filter(user=self.local_user))
- self.assertEqual(
- models.Notification.objects.get().notification_type, 'REPLY')
-
+ self.assertTrue(models.Notification.objects.filter(user=self.local_user))
+ self.assertEqual(models.Notification.objects.get().notification_type, "REPLY")
def test_handle_create_list(self):
- ''' a new list '''
+ """ a new list """
activity = self.create_json
- activity['object'] = {
+ activity["object"] = {
"id": "https://example.com/list/22",
"type": "BookList",
"totalItems": 1,
@@ -218,44 +205,38 @@ class Inbox(TestCase):
"last": "https://example.com/list/22?page=1",
"name": "Test List",
"owner": "https://example.com/user/mouse",
- "to": [
- "https://www.w3.org/ns/activitystreams#Public"
- ],
- "cc": [
- "https://example.com/user/mouse/followers"
- ],
+ "to": ["https://www.w3.org/ns/activitystreams#Public"],
+ "cc": ["https://example.com/user/mouse/followers"],
"summary": "summary text",
"curation": "curated",
- "@context": "https://www.w3.org/ns/activitystreams"
+ "@context": "https://www.w3.org/ns/activitystreams",
}
views.inbox.activity_task(activity)
book_list = models.List.objects.get()
- self.assertEqual(book_list.name, 'Test List')
- self.assertEqual(book_list.curation, 'curated')
- self.assertEqual(book_list.description, 'summary text')
- self.assertEqual(book_list.remote_id, 'https://example.com/list/22')
-
+ self.assertEqual(book_list.name, "Test List")
+ self.assertEqual(book_list.curation, "curated")
+ self.assertEqual(book_list.description, "summary text")
+ self.assertEqual(book_list.remote_id, "https://example.com/list/22")
def test_handle_follow_x(self):
- ''' remote user wants to follow local user '''
+ """ remote user wants to follow local user """
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"
+ "object": "https://example.com/user/mouse",
}
self.assertFalse(models.UserFollowRequest.objects.exists())
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
- as mock:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay") as mock:
views.inbox.activity_task(activity)
self.assertEqual(mock.call_count, 1)
# notification created
notification = models.Notification.objects.get()
self.assertEqual(notification.user, self.local_user)
- self.assertEqual(notification.notification_type, 'FOLLOW')
+ self.assertEqual(notification.notification_type, "FOLLOW")
# the request should have been deleted
self.assertFalse(models.UserFollowRequest.objects.exists())
@@ -264,27 +245,26 @@ class Inbox(TestCase):
follow = models.UserFollows.objects.get(user_object=self.local_user)
self.assertEqual(follow.user_subject, self.remote_user)
-
def test_handle_follow_manually_approved(self):
- ''' needs approval before following '''
+ """ needs approval before following """
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"
+ "object": "https://example.com/user/mouse",
}
self.local_user.manually_approves_followers = True
self.local_user.save(broadcast=False)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
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')
+ self.assertEqual(notification.notification_type, "FOLLOW_REQUEST")
# the request should exist
request = models.UserFollowRequest.objects.get()
@@ -295,38 +275,36 @@ class Inbox(TestCase):
follow = models.UserFollows.objects.all()
self.assertEqual(list(follow), [])
-
def test_handle_unfollow(self):
- ''' remove a relationship '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ remove a relationship """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
rel = models.UserFollows.objects.create(
- user_subject=self.remote_user, user_object=self.local_user)
+ 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,
+ "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"
- }
+ "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())
-
def test_handle_follow_accept(self):
- ''' a remote user approved a follow request from local '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ a remote user approved a follow request from local """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
rel = models.UserFollowRequest.objects.create(
- user_subject=self.local_user,
- user_object=self.remote_user
+ user_subject=self.local_user, user_object=self.remote_user
)
activity = {
"@context": "https://www.w3.org/ns/activitystreams",
@@ -337,8 +315,8 @@ class Inbox(TestCase):
"id": rel.remote_id,
"type": "Follow",
"actor": "https://example.com/user/mouse",
- "object": "https://example.com/users/rat"
- }
+ "object": "https://example.com/users/rat",
+ },
}
self.assertEqual(models.UserFollowRequest.objects.count(), 1)
@@ -353,13 +331,11 @@ class Inbox(TestCase):
self.assertEqual(follows.count(), 1)
self.assertEqual(follows.first(), self.local_user)
-
def test_handle_follow_reject(self):
- ''' turn down a follow request '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ turn down a follow request """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
rel = models.UserFollowRequest.objects.create(
- user_subject=self.local_user,
- user_object=self.remote_user
+ user_subject=self.local_user, user_object=self.remote_user
)
activity = {
"@context": "https://www.w3.org/ns/activitystreams",
@@ -370,8 +346,8 @@ class Inbox(TestCase):
"id": rel.remote_id,
"type": "Follow",
"actor": "https://example.com/user/mouse",
- "object": "https://example.com/users/rat"
- }
+ "object": "https://example.com/users/rat",
+ },
}
self.assertEqual(models.UserFollowRequest.objects.count(), 1)
@@ -382,18 +358,19 @@ class Inbox(TestCase):
self.assertFalse(models.UserFollowRequest.objects.exists())
self.assertFalse(self.remote_user.followers.exists())
-
def test_handle_update_list(self):
- ''' a new list '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ a new list """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
book_list = models.List.objects.create(
- name='hi', remote_id='https://example.com/list/22',
- user=self.local_user)
+ name="hi", remote_id="https://example.com/list/22", user=self.local_user
+ )
activity = {
- 'type': 'Update',
- 'to': [], 'cc': [], 'actor': 'hi',
- 'id': 'sdkjf',
- 'object': {
+ "type": "Update",
+ "to": [],
+ "cc": [],
+ "actor": "hi",
+ "id": "sdkjf",
+ "object": {
"id": "https://example.com/list/22",
"type": "BookList",
"totalItems": 1,
@@ -401,38 +378,33 @@ class Inbox(TestCase):
"last": "https://example.com/list/22?page=1",
"name": "Test List",
"owner": "https://example.com/user/mouse",
- "to": [
- "https://www.w3.org/ns/activitystreams#Public"
- ],
- "cc": [
- "https://example.com/user/mouse/followers"
- ],
+ "to": ["https://www.w3.org/ns/activitystreams#Public"],
+ "cc": ["https://example.com/user/mouse/followers"],
"summary": "summary text",
"curation": "curated",
- "@context": "https://www.w3.org/ns/activitystreams"
- }
+ "@context": "https://www.w3.org/ns/activitystreams",
+ },
}
views.inbox.activity_task(activity)
book_list.refresh_from_db()
- self.assertEqual(book_list.name, 'Test List')
- self.assertEqual(book_list.curation, 'curated')
- self.assertEqual(book_list.description, 'summary text')
- self.assertEqual(book_list.remote_id, 'https://example.com/list/22')
-
+ self.assertEqual(book_list.name, "Test List")
+ self.assertEqual(book_list.curation, "curated")
+ self.assertEqual(book_list.description, "summary text")
+ self.assertEqual(book_list.remote_id, "https://example.com/list/22")
def test_handle_delete_status(self):
- ''' remove a status '''
+ """ remove a status """
self.status.user = self.remote_user
self.status.save(broadcast=False)
self.assertFalse(self.status.deleted)
activity = {
- 'type': 'Delete',
+ "type": "Delete",
"to": ["https://www.w3.org/ns/activitystreams#Public"],
"cc": ["https://example.com/user/mouse/followers"],
- 'id': '%s/activity' % self.status.remote_id,
- 'actor': self.remote_user.remote_id,
- 'object': {'id': self.status.remote_id, 'type': 'Tombstone'},
+ "id": "%s/activity" % self.status.remote_id,
+ "actor": self.remote_user.remote_id,
+ "object": {"id": self.status.remote_id, "type": "Tombstone"},
}
views.inbox.activity_task(activity)
# deletion doens't remove the status, it turns it into a tombstone
@@ -440,30 +412,28 @@ class Inbox(TestCase):
self.assertTrue(status.deleted)
self.assertIsInstance(status.deleted_date, datetime)
-
def test_handle_delete_status_notifications(self):
- ''' remove a status with related notifications '''
+ """ remove a status with related notifications """
self.status.user = self.remote_user
self.status.save(broadcast=False)
models.Notification.objects.create(
related_status=self.status,
user=self.local_user,
- notification_type='MENTION'
+ notification_type="MENTION",
)
# this one is innocent, don't delete it
notif = models.Notification.objects.create(
- user=self.local_user,
- notification_type='MENTION'
+ user=self.local_user, notification_type="MENTION"
)
self.assertFalse(self.status.deleted)
self.assertEqual(models.Notification.objects.count(), 2)
activity = {
- 'type': 'Delete',
+ "type": "Delete",
"to": ["https://www.w3.org/ns/activitystreams#Public"],
"cc": ["https://example.com/user/mouse/followers"],
- 'id': '%s/activity' % self.status.remote_id,
- 'actor': self.remote_user.remote_id,
- 'object': {'id': self.status.remote_id, 'type': 'Tombstone'},
+ "id": "%s/activity" % self.status.remote_id,
+ "actor": self.remote_user.remote_id,
+ "object": {"id": self.status.remote_id, "type": "Tombstone"},
}
views.inbox.activity_task(activity)
# deletion doens't remove the status, it turns it into a tombstone
@@ -475,63 +445,76 @@ class Inbox(TestCase):
self.assertEqual(models.Notification.objects.count(), 1)
self.assertEqual(models.Notification.objects.get(), notif)
-
def test_handle_favorite(self):
- ''' fav a status '''
+ """ fav a status """
activity = {
- '@context': 'https://www.w3.org/ns/activitystreams',
- 'id': 'https://example.com/fav/1',
- 'actor': 'https://example.com/users/rat',
- 'type': 'Like',
- 'published': 'Mon, 25 May 2020 19:31:20 GMT',
- 'object': 'https://example.com/status/1',
+ "@context": "https://www.w3.org/ns/activitystreams",
+ "id": "https://example.com/fav/1",
+ "actor": "https://example.com/users/rat",
+ "type": "Like",
+ "published": "Mon, 25 May 2020 19:31:20 GMT",
+ "object": self.status.remote_id,
}
views.inbox.activity_task(activity)
- fav = models.Favorite.objects.get(remote_id='https://example.com/fav/1')
+ fav = models.Favorite.objects.get(remote_id="https://example.com/fav/1")
self.assertEqual(fav.status, self.status)
- self.assertEqual(fav.remote_id, 'https://example.com/fav/1')
+ self.assertEqual(fav.remote_id, "https://example.com/fav/1")
self.assertEqual(fav.user, self.remote_user)
- def test_handle_unfavorite(self):
- ''' fav a status '''
+ def test_ignore_favorite(self):
+ """ don't try to save an unknown status """
activity = {
- 'id': 'https://example.com/fav/1#undo',
- 'type': 'Undo',
+ "@context": "https://www.w3.org/ns/activitystreams",
+ "id": "https://example.com/fav/1",
+ "actor": "https://example.com/users/rat",
+ "type": "Like",
+ "published": "Mon, 25 May 2020 19:31:20 GMT",
+ "object": "https://unknown.status/not-found",
+ }
+
+ views.inbox.activity_task(activity)
+
+ self.assertFalse(models.Favorite.objects.exists())
+
+ def test_handle_unfavorite(self):
+ """ fav a status """
+ activity = {
+ "id": "https://example.com/fav/1#undo",
+ "type": "Undo",
"to": ["https://www.w3.org/ns/activitystreams#Public"],
"cc": ["https://example.com/user/mouse/followers"],
- 'actor': self.remote_user.remote_id,
- 'object': {
- '@context': 'https://www.w3.org/ns/activitystreams',
- 'id': 'https://example.com/fav/1',
- 'actor': 'https://example.com/users/rat',
- 'type': 'Like',
- 'published': 'Mon, 25 May 2020 19:31:20 GMT',
- 'object': self.status.remote_id,
- }
+ "actor": self.remote_user.remote_id,
+ "object": {
+ "@context": "https://www.w3.org/ns/activitystreams",
+ "id": "https://example.com/fav/1",
+ "actor": "https://example.com/users/rat",
+ "type": "Like",
+ "published": "Mon, 25 May 2020 19:31:20 GMT",
+ "object": self.status.remote_id,
+ },
}
models.Favorite.objects.create(
status=self.status,
user=self.remote_user,
- remote_id='https://example.com/fav/1')
+ remote_id="https://example.com/fav/1",
+ )
self.assertEqual(models.Favorite.objects.count(), 1)
views.inbox.activity_task(activity)
self.assertEqual(models.Favorite.objects.count(), 0)
-
def test_handle_boost(self):
- ''' boost a status '''
+ """ boost a status """
self.assertEqual(models.Notification.objects.count(), 0)
activity = {
- 'type': 'Announce',
- 'id': '%s/boost' % self.status.remote_id,
- 'actor': self.remote_user.remote_id,
- 'object': self.status.remote_id,
+ "type": "Announce",
+ "id": "%s/boost" % self.status.remote_id,
+ "actor": self.remote_user.remote_id,
+ "object": self.status.remote_id,
}
- with patch('bookwyrm.models.status.Status.ignore_activity') \
- as discarder:
+ with patch("bookwyrm.models.status.Status.ignore_activity") as discarder:
discarder.return_value = False
views.inbox.activity_task(activity)
boost = models.Boost.objects.get()
@@ -540,59 +523,56 @@ class Inbox(TestCase):
self.assertEqual(notification.user, self.local_user)
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 '''
+ """ test a boost of a mastodon status that will be discarded """
status = models.Status(
- content='hi',
+ content="hi",
user=self.remote_user,
)
status.save(broadcast=False)
activity = {
- 'type': 'Announce',
- 'id': 'http://www.faraway.com/boost/12',
- 'actor': self.remote_user.remote_id,
- 'object': status.remote_id,
+ "type": "Announce",
+ "id": "http://www.faraway.com/boost/12",
+ "actor": self.remote_user.remote_id,
+ "object": status.remote_id,
}
responses.add(
- responses.GET,
- status.remote_id,
- json=status.to_activity(),
- status=200)
+ responses.GET, status.remote_id, json=status.to_activity(), status=200
+ )
views.inbox.activity_task(activity)
self.assertEqual(models.Boost.objects.count(), 0)
-
def test_handle_unboost(self):
- ''' undo a boost '''
+ """ undo a boost """
boost = models.Boost.objects.create(
- boosted_status=self.status, user=self.remote_user)
+ boosted_status=self.status, user=self.remote_user
+ )
activity = {
- 'type': 'Undo',
- 'actor': 'hi',
- 'id': 'bleh',
+ "type": "Undo",
+ "actor": "hi",
+ "id": "bleh",
"to": ["https://www.w3.org/ns/activitystreams#public"],
"cc": ["https://example.com/user/mouse/followers"],
- 'object': {
- 'type': 'Announce',
- 'id': boost.remote_id,
- 'actor': self.remote_user.remote_id,
- 'object': self.status.remote_id,
- }
+ "object": {
+ "type": "Announce",
+ "id": boost.remote_id,
+ "actor": self.remote_user.remote_id,
+ "object": self.status.remote_id,
+ },
}
views.inbox.activity_task(activity)
-
def test_handle_add_book_to_shelf(self):
- ''' shelving a book '''
- work = models.Work.objects.create(title='work title')
+ """ shelving a book """
+ work = models.Work.objects.create(title="work title")
book = models.Edition.objects.create(
- title='Test', remote_id='https://bookwyrm.social/book/37292',
- parent_work=work)
- shelf = models.Shelf.objects.create(
- user=self.remote_user, name='Test Shelf')
- shelf.remote_id = 'https://bookwyrm.social/user/mouse/shelf/to-read'
+ title="Test",
+ remote_id="https://bookwyrm.social/book/37292",
+ parent_work=work,
+ )
+ shelf = models.Shelf.objects.create(user=self.remote_user, name="Test Shelf")
+ shelf.remote_id = "https://bookwyrm.social/user/mouse/shelf/to-read"
shelf.save()
activity = {
@@ -606,23 +586,24 @@ class Inbox(TestCase):
"id": "https://bookwyrm.social/book/37292",
},
"target": "https://bookwyrm.social/user/mouse/shelf/to-read",
- "@context": "https://www.w3.org/ns/activitystreams"
+ "@context": "https://www.w3.org/ns/activitystreams",
}
views.inbox.activity_task(activity)
self.assertEqual(shelf.books.first(), book)
-
@responses.activate
def test_handle_add_book_to_list(self):
- ''' listing a book '''
- work = models.Work.objects.create(title='work title')
+ """ listing a book """
+ work = models.Work.objects.create(title="work title")
book = models.Edition.objects.create(
- title='Test', remote_id='https://bookwyrm.social/book/37292',
- parent_work=work)
+ title="Test",
+ remote_id="https://bookwyrm.social/book/37292",
+ parent_work=work,
+ )
responses.add(
responses.GET,
- 'https://bookwyrm.social/user/mouse/list/to-read',
+ "https://bookwyrm.social/user/mouse/list/to-read",
json={
"id": "https://example.com/list/22",
"type": "BookList",
@@ -631,16 +612,12 @@ class Inbox(TestCase):
"last": "https://example.com/list/22?page=1",
"name": "Test List",
"owner": "https://example.com/user/mouse",
- "to": [
- "https://www.w3.org/ns/activitystreams#Public"
- ],
- "cc": [
- "https://example.com/user/mouse/followers"
- ],
+ "to": ["https://www.w3.org/ns/activitystreams#Public"],
+ "cc": ["https://example.com/user/mouse/followers"],
"summary": "summary text",
"curation": "curated",
- "@context": "https://www.w3.org/ns/activitystreams"
- }
+ "@context": "https://www.w3.org/ns/activitystreams",
+ },
)
activity = {
@@ -654,26 +631,27 @@ class Inbox(TestCase):
"id": "https://bookwyrm.social/book/37292",
},
"target": "https://bookwyrm.social/user/mouse/list/to-read",
- "@context": "https://www.w3.org/ns/activitystreams"
+ "@context": "https://www.w3.org/ns/activitystreams",
}
views.inbox.activity_task(activity)
booklist = models.List.objects.get()
- self.assertEqual(booklist.name, 'Test List')
+ self.assertEqual(booklist.name, "Test List")
self.assertEqual(booklist.books.first(), book)
-
@responses.activate
def test_handle_tag_book(self):
- ''' listing a book '''
- work = models.Work.objects.create(title='work title')
+ """ listing a book """
+ work = models.Work.objects.create(title="work title")
book = models.Edition.objects.create(
- title='Test', remote_id='https://bookwyrm.social/book/37292',
- parent_work=work)
+ title="Test",
+ remote_id="https://bookwyrm.social/book/37292",
+ parent_work=work,
+ )
responses.add(
responses.GET,
- 'https://www.example.com/tag/cool-tag',
+ "https://www.example.com/tag/cool-tag",
json={
"id": "https://1b1a78582461.ngrok.io/tag/tag",
"type": "OrderedCollection",
@@ -681,8 +659,8 @@ class Inbox(TestCase):
"first": "https://1b1a78582461.ngrok.io/tag/tag?page=1",
"last": "https://1b1a78582461.ngrok.io/tag/tag?page=1",
"name": "cool tag",
- "@context": "https://www.w3.org/ns/activitystreams"
- }
+ "@context": "https://www.w3.org/ns/activitystreams",
+ },
)
activity = {
@@ -696,95 +674,101 @@ class Inbox(TestCase):
"id": "https://bookwyrm.social/book/37292",
},
"target": "https://www.example.com/tag/cool-tag",
- "@context": "https://www.w3.org/ns/activitystreams"
+ "@context": "https://www.w3.org/ns/activitystreams",
}
views.inbox.activity_task(activity)
tag = models.Tag.objects.get()
self.assertFalse(models.List.objects.exists())
- self.assertEqual(tag.name, 'cool tag')
+ self.assertEqual(tag.name, "cool tag")
self.assertEqual(tag.books.first(), book)
-
def test_handle_update_user(self):
- ''' update an existing user '''
+ """ update an existing user """
# we only do this with remote users
self.local_user.local = False
self.local_user.save()
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/ap_user.json')
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/ap_user.json")
userdata = json.loads(datafile.read_bytes())
- del userdata['icon']
+ del userdata["icon"]
self.assertIsNone(self.local_user.name)
- views.inbox.activity_task({
- 'type': 'Update',
- 'to': [], 'cc': [], 'actor': 'hi',
- 'id': 'sdkjf',
- 'object': userdata
- })
+ views.inbox.activity_task(
+ {
+ "type": "Update",
+ "to": [],
+ "cc": [],
+ "actor": "hi",
+ "id": "sdkjf",
+ "object": userdata,
+ }
+ )
user = models.User.objects.get(id=self.local_user.id)
- self.assertEqual(user.name, 'MOUSE?? MOUSE!!')
- self.assertEqual(user.username, 'mouse@example.com')
- self.assertEqual(user.localname, 'mouse')
-
+ self.assertEqual(user.name, "MOUSE?? MOUSE!!")
+ self.assertEqual(user.username, "mouse@example.com")
+ self.assertEqual(user.localname, "mouse")
def test_handle_update_edition(self):
- ''' update an existing edition '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/bw_edition.json')
+ """ update an existing edition """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/bw_edition.json")
bookdata = json.loads(datafile.read_bytes())
models.Work.objects.create(
- title='Test Work', remote_id='https://bookwyrm.social/book/5988')
+ title="Test Work", remote_id="https://bookwyrm.social/book/5988"
+ )
book = models.Edition.objects.create(
- title='Test Book', remote_id='https://bookwyrm.social/book/5989')
+ title="Test Book", remote_id="https://bookwyrm.social/book/5989"
+ )
- del bookdata['authors']
- self.assertEqual(book.title, 'Test Book')
+ del bookdata["authors"]
+ self.assertEqual(book.title, "Test Book")
- with patch(
- 'bookwyrm.activitypub.base_activity.set_related_field.delay'):
- views.inbox.activity_task({
- 'type': 'Update',
- 'to': [], 'cc': [], 'actor': 'hi',
- 'id': 'sdkjf',
- 'object': bookdata
- })
+ with patch("bookwyrm.activitypub.base_activity.set_related_field.delay"):
+ views.inbox.activity_task(
+ {
+ "type": "Update",
+ "to": [],
+ "cc": [],
+ "actor": "hi",
+ "id": "sdkjf",
+ "object": bookdata,
+ }
+ )
book = models.Edition.objects.get(id=book.id)
- self.assertEqual(book.title, 'Piranesi')
-
+ self.assertEqual(book.title, "Piranesi")
def test_handle_update_work(self):
- ''' update an existing edition '''
- datafile = pathlib.Path(__file__).parent.joinpath(
- '../data/bw_work.json')
+ """ update an existing edition """
+ datafile = pathlib.Path(__file__).parent.joinpath("../data/bw_work.json")
bookdata = json.loads(datafile.read_bytes())
book = models.Work.objects.create(
- title='Test Book', remote_id='https://bookwyrm.social/book/5988')
+ title="Test Book", remote_id="https://bookwyrm.social/book/5988"
+ )
- del bookdata['authors']
- self.assertEqual(book.title, 'Test Book')
- with patch(
- 'bookwyrm.activitypub.base_activity.set_related_field.delay'):
- views.inbox.activity_task({
- 'type': 'Update',
- 'to': [], 'cc': [], 'actor': 'hi',
- 'id': 'sdkjf',
- 'object': bookdata
- })
+ del bookdata["authors"]
+ self.assertEqual(book.title, "Test Book")
+ with patch("bookwyrm.activitypub.base_activity.set_related_field.delay"):
+ views.inbox.activity_task(
+ {
+ "type": "Update",
+ "to": [],
+ "cc": [],
+ "actor": "hi",
+ "id": "sdkjf",
+ "object": bookdata,
+ }
+ )
book = models.Work.objects.get(id=book.id)
- self.assertEqual(book.title, 'Piranesi')
-
+ self.assertEqual(book.title, "Piranesi")
def test_handle_blocks(self):
- ''' create a "block" database entry from an activity '''
+ """ create a "block" database entry from an activity """
self.local_user.followers.add(self.remote_user)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.UserFollowRequest.objects.create(
- user_subject=self.local_user,
- user_object=self.remote_user)
+ user_subject=self.local_user, user_object=self.remote_user
+ )
self.assertTrue(models.UserFollows.objects.exists())
self.assertTrue(models.UserFollowRequest.objects.exists())
@@ -793,42 +777,41 @@ class Inbox(TestCase):
"id": "https://example.com/9e1f41ac-9ddd-4159",
"type": "Block",
"actor": "https://example.com/users/rat",
- "object": "https://example.com/user/mouse"
+ "object": "https://example.com/user/mouse",
}
views.inbox.activity_task(activity)
block = models.UserBlocks.objects.get()
self.assertEqual(block.user_subject, self.remote_user)
self.assertEqual(block.user_object, self.local_user)
- self.assertEqual(
- block.remote_id, 'https://example.com/9e1f41ac-9ddd-4159')
+ self.assertEqual(block.remote_id, "https://example.com/9e1f41ac-9ddd-4159")
self.assertFalse(models.UserFollows.objects.exists())
self.assertFalse(models.UserFollowRequest.objects.exists())
-
def test_handle_unblock(self):
- ''' unblock a user '''
+ """ unblock a user """
self.remote_user.blocks.add(self.local_user)
block = models.UserBlocks.objects.get()
- block.remote_id = 'https://example.com/9e1f41ac-9ddd-4159'
+ block.remote_id = "https://example.com/9e1f41ac-9ddd-4159"
block.save()
self.assertEqual(block.user_subject, self.remote_user)
self.assertEqual(block.user_object, self.local_user)
activity = {
- 'type': 'Undo',
- 'actor': 'hi',
- 'id': 'bleh',
+ "type": "Undo",
+ "actor": "hi",
+ "id": "bleh",
"to": ["https://www.w3.org/ns/activitystreams#public"],
"cc": ["https://example.com/user/mouse/followers"],
- 'object': {
+ "object": {
"@context": "https://www.w3.org/ns/activitystreams",
"id": "https://example.com/9e1f41ac-9ddd-4159",
"type": "Block",
"actor": "https://example.com/users/rat",
- "object": "https://example.com/user/mouse"
- }}
+ "object": "https://example.com/user/mouse",
+ },
+ }
views.inbox.activity_task(activity)
self.assertFalse(models.UserBlocks.objects.exists())
diff --git a/bookwyrm/tests/views/test_interaction.py b/bookwyrm/tests/views/test_interaction.py
index c6d39f29..857f7061 100644
--- a/bookwyrm/tests/views/test_interaction.py
+++ b/bookwyrm/tests/views/test_interaction.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.test import TestCase
from django.test.client import RequestFactory
@@ -7,40 +7,44 @@ from bookwyrm import models, views
class InteractionViews(TestCase):
- ''' viewing and creating statuses '''
+ """ viewing and creating statuses """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- with patch('bookwyrm.models.user.set_remote_server'):
+ with patch("bookwyrm.models.user.set_remote_server"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@email.com', 'ratword',
+ "rat",
+ "rat@email.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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
- work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=work,
)
-
def test_handle_favorite(self):
- ''' create and broadcast faving a status '''
+ """ create and broadcast faving a status """
view = views.Favorite.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.remote_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
view(request, status.id)
fav = models.Favorite.objects.get()
@@ -48,106 +52,100 @@ class InteractionViews(TestCase):
self.assertEqual(fav.user, self.remote_user)
notification = models.Notification.objects.get()
- self.assertEqual(notification.notification_type, 'FAVORITE')
+ self.assertEqual(notification.notification_type, "FAVORITE")
self.assertEqual(notification.user, self.local_user)
self.assertEqual(notification.related_user, self.remote_user)
-
def test_handle_unfavorite(self):
- ''' unfav a status '''
+ """ unfav a status """
view = views.Unfavorite.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.remote_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
views.Favorite.as_view()(request, status.id)
self.assertEqual(models.Favorite.objects.count(), 1)
self.assertEqual(models.Notification.objects.count(), 1)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, status.id)
self.assertEqual(models.Favorite.objects.count(), 0)
self.assertEqual(models.Notification.objects.count(), 0)
-
def test_handle_boost(self):
- ''' boost a status '''
+ """ boost a status """
view = views.Boost.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.remote_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
view(request, status.id)
boost = models.Boost.objects.get()
self.assertEqual(boost.boosted_status, status)
self.assertEqual(boost.user, self.remote_user)
- self.assertEqual(boost.privacy, 'public')
+ self.assertEqual(boost.privacy, "public")
notification = models.Notification.objects.get()
- self.assertEqual(notification.notification_type, 'BOOST')
+ self.assertEqual(notification.notification_type, "BOOST")
self.assertEqual(notification.user, self.local_user)
self.assertEqual(notification.related_user, self.remote_user)
self.assertEqual(notification.related_status, status)
def test_handle_boost_unlisted(self):
- ''' boost a status '''
+ """ boost a status """
view = views.Boost.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
status = models.Status.objects.create(
- user=self.local_user, content='hi', privacy='unlisted')
+ user=self.local_user, content="hi", privacy="unlisted"
+ )
view(request, status.id)
boost = models.Boost.objects.get()
- self.assertEqual(boost.privacy, 'unlisted')
+ self.assertEqual(boost.privacy, "unlisted")
def test_handle_boost_private(self):
- ''' boost a status '''
+ """ boost a status """
view = views.Boost.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
status = models.Status.objects.create(
- user=self.local_user, content='hi', privacy='followers')
+ user=self.local_user, content="hi", privacy="followers"
+ )
view(request, status.id)
self.assertFalse(models.Boost.objects.exists())
def test_handle_boost_twice(self):
- ''' boost a status '''
+ """ boost a status """
view = views.Boost.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
view(request, status.id)
view(request, status.id)
self.assertEqual(models.Boost.objects.count(), 1)
-
def test_handle_unboost(self):
- ''' undo a boost '''
+ """ undo a boost """
view = views.Unboost.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
views.Boost.as_view()(request, status.id)
self.assertEqual(models.Boost.objects.count(), 1)
self.assertEqual(models.Notification.objects.count(), 1)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
- as mock:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay") as mock:
view(request, status.id)
self.assertEqual(mock.call_count, 1)
self.assertEqual(models.Boost.objects.count(), 0)
diff --git a/bookwyrm/tests/views/test_invite.py b/bookwyrm/tests/views/test_invite.py
index e93e7209..cd2276c0 100644
--- a/bookwyrm/tests/views/test_invite.py
+++ b/bookwyrm/tests/views/test_invite.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import AnonymousUser
@@ -11,36 +11,39 @@ from bookwyrm import views
class InviteViews(TestCase):
- ''' every response to a get request, html or json '''
+ """ every response to a get request, html or json """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
models.SiteSettings.objects.create()
-
def test_invite_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Invite.as_view()
- models.SiteInvite.objects.create(code='hi', user=self.local_user)
- request = self.factory.get('')
+ models.SiteInvite.objects.create(code="hi", user=self.local_user)
+ request = self.factory.get("")
request.user = AnonymousUser
# why?? this is annoying.
request.user.is_authenticated = False
- with patch('bookwyrm.models.site.SiteInvite.valid') as invite:
+ with patch("bookwyrm.models.site.SiteInvite.valid") as invite:
invite.return_value = True
- result = view(request, 'hi')
+ result = view(request, "hi")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_manage_invites(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.ManageInvites.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
request.user.is_superuser = True
result = view(request)
diff --git a/bookwyrm/tests/views/test_isbn.py b/bookwyrm/tests/views/test_isbn.py
new file mode 100644
index 00000000..c7ae1f39
--- /dev/null
+++ b/bookwyrm/tests/views/test_isbn.py
@@ -0,0 +1,53 @@
+""" test for app action functionality """
+import json
+from unittest.mock import patch
+
+from django.http import JsonResponse
+from django.template.response import TemplateResponse
+from django.test import TestCase
+from django.test.client import RequestFactory
+
+from bookwyrm import models, views
+from bookwyrm.connectors import abstract_connector
+from bookwyrm.settings import DOMAIN
+
+
+class IsbnViews(TestCase):
+ """ tag views"""
+
+ def setUp(self):
+ """ we need basic test data and mocks """
+ self.factory = RequestFactory()
+ 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",
+ )
+ self.work = models.Work.objects.create(title="Test Work")
+ self.book = models.Edition.objects.create(
+ title="Test Book",
+ isbn_13="1234567890123",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
+ )
+ models.Connector.objects.create(
+ identifier="self", connector_file="self_connector", local=True
+ )
+ models.SiteSettings.objects.create()
+
+ def test_isbn_json_response(self):
+ """ searches local data only and returns book data in json format """
+ view = views.Isbn.as_view()
+ request = self.factory.get("")
+ with patch("bookwyrm.views.isbn.is_api_request") as is_api:
+ is_api.return_value = True
+ response = view(request, isbn="1234567890123")
+ self.assertIsInstance(response, JsonResponse)
+
+ data = json.loads(response.content)
+ self.assertEqual(len(data), 1)
+ self.assertEqual(data[0]["title"], "Test Book")
+ self.assertEqual(data[0]["key"], "https://%s/book/%d" % (DOMAIN, self.book.id))
diff --git a/bookwyrm/tests/views/test_landing.py b/bookwyrm/tests/views/test_landing.py
index 5e0e50cf..910e4a85 100644
--- a/bookwyrm/tests/views/test_landing.py
+++ b/bookwyrm/tests/views/test_landing.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from django.contrib.auth.models import AnonymousUser
from django.template.response import TemplateResponse
from django.test import TestCase
@@ -9,22 +9,26 @@ from bookwyrm import views
class LandingViews(TestCase):
- ''' pages you land on without really trying '''
+ """ pages you land on without really trying """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
models.SiteSettings.objects.create()
-
def test_home_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Home.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertEqual(result.status_code, 200)
@@ -36,21 +40,19 @@ class LandingViews(TestCase):
self.assertEqual(result.status_code, 200)
result.render()
-
def test_about_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.About.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_discover(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Discover.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
result = view(request)
self.assertIsInstance(result, TemplateResponse)
diff --git a/bookwyrm/tests/views/test_list.py b/bookwyrm/tests/views/test_list.py
index e41d9806..cc895ad1 100644
--- a/bookwyrm/tests/views/test_list.py
+++ b/bookwyrm/tests/views/test_list.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import AnonymousUser
@@ -9,44 +9,52 @@ from django.test.client import RequestFactory
from bookwyrm import models, views
from bookwyrm.activitypub import ActivitypubResponse
-#pylint: disable=unused-argument
+# pylint: disable=unused-argument
class ListViews(TestCase):
- ''' tag views'''
+ """ tag views"""
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
self.rat = models.User.objects.create_user(
- 'rat@local.com', 'rat@rat.com', 'ratword',
- local=True, localname='rat',
- remote_id='https://example.com/users/rat',
+ "rat@local.com",
+ "rat@rat.com",
+ "ratword",
+ local=True,
+ localname="rat",
+ remote_id="https://example.com/users/rat",
)
- work = models.Work.objects.create(title='Work')
+ work = models.Work.objects.create(title="Work")
self.book = models.Edition.objects.create(
- title='Example Edition',
- remote_id='https://example.com/book/1',
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
parent_work=work,
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.list = models.List.objects.create(
- name='Test List', user=self.local_user)
+ name="Test List", user=self.local_user
+ )
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
models.SiteSettings.objects.create()
-
def test_lists_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Lists.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- models.List.objects.create(name='Public list', user=self.local_user)
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ models.List.objects.create(name="Public list", user=self.local_user)
models.List.objects.create(
- name='Private list', privacy='direct', user=self.local_user)
- request = self.factory.get('')
+ name="Private list", privacy="direct", user=self.local_user
+ )
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
@@ -61,42 +69,45 @@ class ListViews(TestCase):
result.render()
self.assertEqual(result.status_code, 200)
-
def test_lists_create(self):
- ''' create list view '''
+ """ create list view """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Create')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
+ self.assertEqual(activity["type"], "Create")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+
models.List.broadcast = mock_broadcast
view = views.Lists.as_view()
- request = self.factory.post('', {
- 'name': 'A list',
- 'description': 'wow',
- 'privacy': 'unlisted',
- 'curation': 'open',
- 'user': self.local_user.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "name": "A list",
+ "description": "wow",
+ "privacy": "unlisted",
+ "curation": "open",
+ "user": self.local_user.id,
+ },
+ )
request.user = self.local_user
result = view(request)
self.assertEqual(result.status_code, 302)
- new_list = models.List.objects.filter(name='A list').get()
- self.assertEqual(new_list.description, 'wow')
- self.assertEqual(new_list.privacy, 'unlisted')
- self.assertEqual(new_list.curation, 'open')
+ new_list = models.List.objects.filter(name="A list").get()
+ self.assertEqual(new_list.description, "wow")
+ self.assertEqual(new_list.privacy, "unlisted")
+ self.assertEqual(new_list.curation, "open")
models.List.broadcast = real_broadcast
-
def test_list_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.List.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.list.is_api_request') as is_api:
+ with patch("bookwyrm.views.list.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.list.id)
self.assertIsInstance(result, TemplateResponse)
@@ -104,68 +115,72 @@ class ListViews(TestCase):
self.assertEqual(result.status_code, 200)
request.user = self.anonymous_user
- with patch('bookwyrm.views.list.is_api_request') as is_api:
+ with patch("bookwyrm.views.list.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.list.id)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.list.is_api_request') as is_api:
+ with patch("bookwyrm.views.list.is_api_request") as is_api:
is_api.return_value = True
result = view(request, self.list.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
- request = self.factory.get('/?page=1')
+ request = self.factory.get("/?page=1")
request.user = self.local_user
- with patch('bookwyrm.views.list.is_api_request') as is_api:
+ with patch("bookwyrm.views.list.is_api_request") as is_api:
is_api.return_value = True
result = view(request, self.list.id)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_list_edit(self):
- ''' edit a list '''
+ """ edit a list """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Update')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['object']['id'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Update")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["object"]["id"], self.list.remote_id)
+
models.List.broadcast = mock_broadcast
view = views.List.as_view()
- request = self.factory.post('', {
- 'name': 'New Name',
- 'description': 'wow',
- 'privacy': 'direct',
- 'curation': 'curated',
- 'user': self.local_user.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "name": "New Name",
+ "description": "wow",
+ "privacy": "direct",
+ "curation": "curated",
+ "user": self.local_user.id,
+ },
+ )
request.user = self.local_user
result = view(request, self.list.id)
self.assertEqual(result.status_code, 302)
self.list.refresh_from_db()
- self.assertEqual(self.list.name, 'New Name')
- self.assertEqual(self.list.description, 'wow')
- self.assertEqual(self.list.privacy, 'direct')
- self.assertEqual(self.list.curation, 'curated')
+ self.assertEqual(self.list.name, "New Name")
+ self.assertEqual(self.list.description, "wow")
+ self.assertEqual(self.list.privacy, "direct")
+ self.assertEqual(self.list.curation, "curated")
models.List.broadcast = real_broadcast
-
def test_curate_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Curate.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- models.List.objects.create(name='Public list', user=self.local_user)
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ models.List.objects.create(name="Public list", user=self.local_user)
models.List.objects.create(
- name='Private list', privacy='direct', user=self.local_user)
- request = self.factory.get('')
+ name="Private list", privacy="direct", user=self.local_user
+ )
+ request = self.factory.get("")
request.user = self.local_user
result = view(request, self.list.id)
@@ -177,31 +192,35 @@ class ListViews(TestCase):
result = view(request, self.list.id)
self.assertEqual(result.status_code, 302)
-
def test_curate_approve(self):
- ''' approve a pending item '''
+ """ approve a pending item """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user, **kwargs):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+
models.ListItem.broadcast = mock_broadcast
view = views.Curate.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
pending = models.ListItem.objects.create(
book_list=self.list,
user=self.local_user,
book=self.book,
- approved=False
+ approved=False,
)
- request = self.factory.post('', {
- 'item': pending.id,
- 'approved': 'true',
- })
+ request = self.factory.post(
+ "",
+ {
+ "item": pending.id,
+ "approved": "true",
+ },
+ )
request.user = self.local_user
view(request, self.list.id)
@@ -211,43 +230,49 @@ class ListViews(TestCase):
self.assertTrue(pending.approved)
models.ListItem.broadcast = real_broadcast
-
def test_curate_reject(self):
- ''' approve a pending item '''
+ """ approve a pending item """
view = views.Curate.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
pending = models.ListItem.objects.create(
book_list=self.list,
user=self.local_user,
book=self.book,
- approved=False
+ approved=False,
)
- request = self.factory.post('', {
- 'item': pending.id,
- 'approved': 'false',
- })
+ request = self.factory.post(
+ "",
+ {
+ "item": pending.id,
+ "approved": "false",
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, self.list.id)
self.assertFalse(self.list.books.exists())
self.assertFalse(models.ListItem.objects.exists())
-
def test_add_book(self):
- ''' put a book on a list '''
+ """ put a book on a list """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+
models.ListItem.broadcast = mock_broadcast
- request = self.factory.post('', {
- 'book': self.book.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "book": self.book.id,
+ },
+ )
request.user = self.local_user
views.list.add_book(request, self.list.id)
@@ -257,22 +282,26 @@ class ListViews(TestCase):
self.assertTrue(item.approved)
models.ListItem.broadcast = real_broadcast
-
def test_add_book_outsider(self):
- ''' put a book on a list '''
+ """ put a book on a list """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.rat.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.rat.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.rat.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+
models.ListItem.broadcast = mock_broadcast
- self.list.curation = 'open'
+ self.list.curation = "open"
self.list.save(broadcast=False)
- request = self.factory.post('', {
- 'book': self.book.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "book": self.book.id,
+ },
+ )
request.user = self.rat
views.list.add_book(request, self.list.id)
@@ -282,23 +311,27 @@ class ListViews(TestCase):
self.assertTrue(item.approved)
models.ListItem.broadcast = real_broadcast
-
def test_add_book_pending(self):
- ''' put a book on a list awaiting approval '''
+ """ put a book on a list awaiting approval """
real_broadcast = models.List.broadcast
+
def mock_broadcast(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.rat.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.rat.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
- self.assertEqual(activity['object']['id'], self.book.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.rat.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+ self.assertEqual(activity["object"]["id"], self.book.remote_id)
+
models.ListItem.broadcast = mock_broadcast
- self.list.curation = 'curated'
+ self.list.curation = "curated"
self.list.save(broadcast=False)
- request = self.factory.post('', {
- 'book': self.book.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "book": self.book.id,
+ },
+ )
request.user = self.rat
views.list.add_book(request, self.list.id)
@@ -308,23 +341,27 @@ class ListViews(TestCase):
self.assertFalse(item.approved)
models.ListItem.broadcast = real_broadcast
-
def test_add_book_self_curated(self):
- ''' put a book on a list automatically approved '''
+ """ put a book on a list automatically approved """
real_broadcast = models.ListItem.broadcast
+
def mock_broadcast(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Add')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Add")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+
models.ListItem.broadcast = mock_broadcast
- self.list.curation = 'curated'
+ self.list.curation = "curated"
self.list.save(broadcast=False)
- request = self.factory.post('', {
- 'book': self.book.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "book": self.book.id,
+ },
+ )
request.user = self.local_user
views.list.add_book(request, self.list.id)
@@ -334,12 +371,11 @@ class ListViews(TestCase):
self.assertTrue(item.approved)
models.ListItem.broadcast = real_broadcast
-
def test_remove_book(self):
- ''' take an item off a list '''
+ """ take an item off a list """
real_broadcast = models.ListItem.broadcast
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
item = models.ListItem.objects.create(
book_list=self.list,
user=self.local_user,
@@ -348,15 +384,19 @@ class ListViews(TestCase):
self.assertTrue(self.list.listitem_set.exists())
def mock_broadcast(_, activity, user):
- ''' ok '''
+ """ ok """
self.assertEqual(user.remote_id, self.local_user.remote_id)
- self.assertEqual(activity['type'], 'Remove')
- self.assertEqual(activity['actor'], self.local_user.remote_id)
- self.assertEqual(activity['target'], self.list.remote_id)
+ self.assertEqual(activity["type"], "Remove")
+ self.assertEqual(activity["actor"], self.local_user.remote_id)
+ self.assertEqual(activity["target"], self.list.remote_id)
+
models.ListItem.broadcast = mock_broadcast
- request = self.factory.post('', {
- 'item': item.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "item": item.id,
+ },
+ )
request.user = self.local_user
views.list.remove_book(request, self.list.id)
@@ -364,19 +404,21 @@ class ListViews(TestCase):
self.assertFalse(self.list.listitem_set.exists())
models.ListItem.broadcast = real_broadcast
-
def test_remove_book_unauthorized(self):
- ''' take an item off a list '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ take an item off a list """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
item = models.ListItem.objects.create(
book_list=self.list,
user=self.local_user,
book=self.book,
)
self.assertTrue(self.list.listitem_set.exists())
- request = self.factory.post('', {
- 'item': item.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "item": item.id,
+ },
+ )
request.user = self.rat
views.list.remove_book(request, self.list.id)
diff --git a/bookwyrm/tests/views/test_notifications.py b/bookwyrm/tests/views/test_notifications.py
index 55513359..6d92485e 100644
--- a/bookwyrm/tests/views/test_notifications.py
+++ b/bookwyrm/tests/views/test_notifications.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from django.template.response import TemplateResponse
from django.test import TestCase
from django.test.client import RequestFactory
@@ -8,19 +8,24 @@ from bookwyrm import views
class NotificationViews(TestCase):
- ''' notifications '''
+ """ notifications """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
models.SiteSettings.objects.create()
def test_notifications_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Notifications.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
@@ -28,14 +33,16 @@ class NotificationViews(TestCase):
self.assertEqual(result.status_code, 200)
def test_clear_notifications(self):
- ''' erase notifications '''
+ """ erase notifications """
models.Notification.objects.create(
- user=self.local_user, notification_type='FAVORITE')
+ user=self.local_user, notification_type="FAVORITE"
+ )
models.Notification.objects.create(
- user=self.local_user, notification_type='MENTION', read=True)
+ user=self.local_user, notification_type="MENTION", read=True
+ )
self.assertEqual(models.Notification.objects.count(), 2)
view = views.Notifications.as_view()
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
result = view(request)
self.assertEqual(result.status_code, 302)
diff --git a/bookwyrm/tests/views/test_outbox.py b/bookwyrm/tests/views/test_outbox.py
index 7986dea6..5934eb7c 100644
--- a/bookwyrm/tests/views/test_outbox.py
+++ b/bookwyrm/tests/views/test_outbox.py
@@ -1,4 +1,4 @@
-''' sending out activities '''
+""" sending out activities """
from unittest.mock import patch
import json
@@ -12,118 +12,127 @@ from bookwyrm.settings import USER_AGENT
# pylint: disable=too-many-public-methods
class OutboxView(TestCase):
- ''' sends out activities '''
+ """ sends out activities """
+
def setUp(self):
- ''' we'll need some data '''
+ """ we'll need some data """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=work,
)
-
def test_outbox(self):
- ''' returns user's statuses '''
- request = self.factory.get('')
- result = views.Outbox.as_view()(request, 'mouse')
+ """ returns user's statuses """
+ request = self.factory.get("")
+ result = views.Outbox.as_view()(request, "mouse")
self.assertIsInstance(result, JsonResponse)
def test_outbox_bad_method(self):
- ''' can't POST to outbox '''
- request = self.factory.post('')
- result = views.Outbox.as_view()(request, 'mouse')
+ """ can't POST to outbox """
+ request = self.factory.post("")
+ result = views.Outbox.as_view()(request, "mouse")
self.assertEqual(result.status_code, 405)
def test_outbox_unknown_user(self):
- ''' should 404 for unknown and remote users '''
- request = self.factory.post('')
- result = views.Outbox.as_view()(request, 'beepboop')
+ """ should 404 for unknown and remote users """
+ request = self.factory.post("")
+ result = views.Outbox.as_view()(request, "beepboop")
self.assertEqual(result.status_code, 405)
- result = views.Outbox.as_view()(request, 'rat')
+ result = views.Outbox.as_view()(request, "rat")
self.assertEqual(result.status_code, 405)
def test_outbox_privacy(self):
- ''' don't show dms et cetera in outbox '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ don't show dms et cetera in outbox """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.Status.objects.create(
- content='PRIVATE!!', user=self.local_user, privacy='direct')
+ content="PRIVATE!!", user=self.local_user, privacy="direct"
+ )
models.Status.objects.create(
- content='bffs ONLY', user=self.local_user, privacy='followers')
+ content="bffs ONLY", user=self.local_user, privacy="followers"
+ )
models.Status.objects.create(
- content='unlisted status', user=self.local_user,
- privacy='unlisted')
+ content="unlisted status", user=self.local_user, privacy="unlisted"
+ )
models.Status.objects.create(
- content='look at this', user=self.local_user, privacy='public')
+ content="look at this", user=self.local_user, privacy="public"
+ )
- request = self.factory.get('')
- result = views.Outbox.as_view()(request, 'mouse')
+ request = self.factory.get("")
+ result = views.Outbox.as_view()(request, "mouse")
self.assertIsInstance(result, JsonResponse)
data = json.loads(result.content)
- self.assertEqual(data['type'], 'OrderedCollection')
- self.assertEqual(data['totalItems'], 2)
+ self.assertEqual(data["type"], "OrderedCollection")
+ self.assertEqual(data["totalItems"], 2)
def test_outbox_filter(self):
- ''' if we only care about reviews, only get reviews '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ if we only care about reviews, only get reviews """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.Review.objects.create(
- content='look at this', name='hi', rating=1,
- book=self.book, user=self.local_user)
- models.Status.objects.create(
- content='look at this', user=self.local_user)
+ content="look at this",
+ name="hi",
+ rating=1,
+ book=self.book,
+ user=self.local_user,
+ )
+ models.Status.objects.create(content="look at this", user=self.local_user)
- request = self.factory.get('', {'type': 'bleh'})
- result = views.Outbox.as_view()(request, 'mouse')
+ request = self.factory.get("", {"type": "bleh"})
+ result = views.Outbox.as_view()(request, "mouse")
self.assertIsInstance(result, JsonResponse)
data = json.loads(result.content)
- self.assertEqual(data['type'], 'OrderedCollection')
- self.assertEqual(data['totalItems'], 2)
+ self.assertEqual(data["type"], "OrderedCollection")
+ self.assertEqual(data["totalItems"], 2)
- request = self.factory.get('', {'type': 'Review'})
- result = views.Outbox.as_view()(request, 'mouse')
+ request = self.factory.get("", {"type": "Review"})
+ result = views.Outbox.as_view()(request, "mouse")
self.assertIsInstance(result, JsonResponse)
data = json.loads(result.content)
- self.assertEqual(data['type'], 'OrderedCollection')
- self.assertEqual(data['totalItems'], 1)
+ self.assertEqual(data["type"], "OrderedCollection")
+ self.assertEqual(data["totalItems"], 1)
def test_outbox_bookwyrm_request_true(self):
- ''' should differentiate between bookwyrm and outside requests '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ should differentiate between bookwyrm and outside requests """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.Review.objects.create(
- name='hi',
- content='look at this',
+ name="hi",
+ content="look at this",
user=self.local_user,
book=self.book,
- privacy='public',
+ privacy="public",
)
- request = self.factory.get('', {'page': 1}, HTTP_USER_AGENT=USER_AGENT)
- result = views.Outbox.as_view()(request, 'mouse')
+ request = self.factory.get("", {"page": 1}, HTTP_USER_AGENT=USER_AGENT)
+ result = views.Outbox.as_view()(request, "mouse")
data = json.loads(result.content)
- self.assertEqual(len(data['orderedItems']), 1)
- self.assertEqual(data['orderedItems'][0]['type'], 'Review')
+ self.assertEqual(len(data["orderedItems"]), 1)
+ self.assertEqual(data["orderedItems"][0]["type"], "Review")
def test_outbox_bookwyrm_request_false(self):
- ''' should differentiate between bookwyrm and outside requests '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ should differentiate between bookwyrm and outside requests """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.Review.objects.create(
- name='hi',
- content='look at this',
+ name="hi",
+ content="look at this",
user=self.local_user,
book=self.book,
- privacy='public',
+ privacy="public",
)
- request = self.factory.get('', {'page': 1})
- result = views.Outbox.as_view()(request, 'mouse')
+ request = self.factory.get("", {"page": 1})
+ result = views.Outbox.as_view()(request, "mouse")
data = json.loads(result.content)
- self.assertEqual(len(data['orderedItems']), 1)
- self.assertEqual(data['orderedItems'][0]['type'], 'Article')
+ self.assertEqual(len(data["orderedItems"]), 1)
+ self.assertEqual(data["orderedItems"][0]["type"], "Article")
diff --git a/bookwyrm/tests/views/test_password.py b/bookwyrm/tests/views/test_password.py
index 9fc37fdb..f67f5538 100644
--- a/bookwyrm/tests/views/test_password.py
+++ b/bookwyrm/tests/views/test_password.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import AnonymousUser
@@ -10,22 +10,26 @@ from bookwyrm import models, views
class PasswordViews(TestCase):
- ''' view user and edit profile '''
+ """ view user and edit profile """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.com', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "password",
+ local=True,
+ localname="mouse",
+ )
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
models.SiteSettings.objects.create(id=1)
-
def test_password_reset_request(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.PasswordResetRequest.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
@@ -33,76 +37,63 @@ class PasswordViews(TestCase):
result.render()
self.assertEqual(result.status_code, 200)
-
def test_password_reset_request_post(self):
- ''' send 'em an email '''
- request = self.factory.post('', {'email': 'aa@bb.ccc'})
+ """ send 'em an email """
+ request = self.factory.post("", {"email": "aa@bb.ccc"})
view = views.PasswordResetRequest.as_view()
resp = view(request)
self.assertEqual(resp.status_code, 302)
- request = self.factory.post('', {'email': 'mouse@mouse.com'})
- with patch('bookwyrm.emailing.send_email.delay'):
+ request = self.factory.post("", {"email": "mouse@mouse.com"})
+ with patch("bookwyrm.emailing.send_email.delay"):
resp = view(request)
resp.render()
- self.assertEqual(
- models.PasswordReset.objects.get().user, self.local_user)
+ self.assertEqual(models.PasswordReset.objects.get().user, self.local_user)
def test_password_reset(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.PasswordReset.as_view()
code = models.PasswordReset.objects.create(user=self.local_user)
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.anonymous_user
result = view(request, code.code)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_password_reset_post(self):
- ''' reset from code '''
+ """ reset from code """
view = views.PasswordReset.as_view()
code = models.PasswordReset.objects.create(user=self.local_user)
- request = self.factory.post('', {
- 'password': 'hi',
- 'confirm-password': 'hi'
- })
- with patch('bookwyrm.views.password.login'):
+ request = self.factory.post("", {"password": "hi", "confirm-password": "hi"})
+ with patch("bookwyrm.views.password.login"):
resp = view(request, code.code)
self.assertEqual(resp.status_code, 302)
self.assertFalse(models.PasswordReset.objects.exists())
def test_password_reset_wrong_code(self):
- ''' reset from code '''
+ """ reset from code """
view = views.PasswordReset.as_view()
models.PasswordReset.objects.create(user=self.local_user)
- request = self.factory.post('', {
- 'password': 'hi',
- 'confirm-password': 'hi'
- })
- resp = view(request, 'jhgdkfjgdf')
+ request = self.factory.post("", {"password": "hi", "confirm-password": "hi"})
+ resp = view(request, "jhgdkfjgdf")
resp.render()
self.assertTrue(models.PasswordReset.objects.exists())
def test_password_reset_mismatch(self):
- ''' reset from code '''
+ """ reset from code """
view = views.PasswordReset.as_view()
code = models.PasswordReset.objects.create(user=self.local_user)
- request = self.factory.post('', {
- 'password': 'hi',
- 'confirm-password': 'hihi'
- })
+ request = self.factory.post("", {"password": "hi", "confirm-password": "hihi"})
resp = view(request, code.code)
resp.render()
self.assertTrue(models.PasswordReset.objects.exists())
-
def test_password_change_get(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.ChangePassword.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
@@ -110,28 +101,21 @@ class PasswordViews(TestCase):
result.render()
self.assertEqual(result.status_code, 200)
-
def test_password_change(self):
- ''' change password '''
+ """ change password """
view = views.ChangePassword.as_view()
password_hash = self.local_user.password
- request = self.factory.post('', {
- 'password': 'hi',
- 'confirm-password': 'hi'
- })
+ request = self.factory.post("", {"password": "hi", "confirm-password": "hi"})
request.user = self.local_user
- with patch('bookwyrm.views.password.login'):
+ with patch("bookwyrm.views.password.login"):
view(request)
self.assertNotEqual(self.local_user.password, password_hash)
def test_password_change_mismatch(self):
- ''' change password '''
+ """ change password """
view = views.ChangePassword.as_view()
password_hash = self.local_user.password
- request = self.factory.post('', {
- 'password': 'hi',
- 'confirm-password': 'hihi'
- })
+ request = self.factory.post("", {"password": "hi", "confirm-password": "hihi"})
request.user = self.local_user
view(request)
self.assertEqual(self.local_user.password, password_hash)
diff --git a/bookwyrm/tests/views/test_reading.py b/bookwyrm/tests/views/test_reading.py
index 2bb052ae..96d1f1f4 100644
--- a/bookwyrm/tests/views/test_reading.py
+++ b/bookwyrm/tests/views/test_reading.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
import dateutil
from django.test import TestCase
@@ -7,45 +7,54 @@ from django.utils import timezone
from bookwyrm import models, views
+
class ReadingViews(TestCase):
- ''' viewing and creating statuses '''
+ """ viewing and creating statuses """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.work = models.Work.objects.create(title='Test Work')
+ self.work = models.Work.objects.create(title="Test Work")
self.book = models.Edition.objects.create(
- title='Test Book',
- remote_id='https://example.com/book/1',
- parent_work=self.work
+ title="Test Book",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
- with patch('bookwyrm.models.user.set_remote_server.delay'):
+ with patch("bookwyrm.models.user.set_remote_server.delay"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'ratword',
+ "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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
-
def test_start_reading(self):
- ''' begin a book '''
- shelf = self.local_user.shelf_set.get(identifier='reading')
+ """ begin a book """
+ shelf = self.local_user.shelf_set.get(identifier="reading")
self.assertFalse(shelf.books.exists())
self.assertFalse(models.Status.objects.exists())
- request = self.factory.post('', {
- 'post-status': True,
- 'privacy': 'followers',
- 'start_date': '2020-01-05',
- })
+ request = self.factory.post(
+ "",
+ {
+ "post-status": True,
+ "privacy": "followers",
+ "start_date": "2020-01-05",
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.start_reading(request, self.book.id)
self.assertEqual(shelf.books.get(), self.book)
@@ -53,7 +62,7 @@ class ReadingViews(TestCase):
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.mention_books.get(), self.book)
- self.assertEqual(status.privacy, 'followers')
+ self.assertEqual(status.privacy, "followers")
readthrough = models.ReadThrough.objects.get()
self.assertIsNotNone(readthrough.start_date)
@@ -61,45 +70,47 @@ class ReadingViews(TestCase):
self.assertEqual(readthrough.user, self.local_user)
self.assertEqual(readthrough.book, self.book)
-
def test_start_reading_reshelf(self):
- ''' begin a book '''
- to_read_shelf = self.local_user.shelf_set.get(identifier='to-read')
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ begin a book """
+ to_read_shelf = self.local_user.shelf_set.get(identifier="to-read")
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.ShelfBook.objects.create(
- shelf=to_read_shelf, book=self.book, user=self.local_user)
- shelf = self.local_user.shelf_set.get(identifier='reading')
+ shelf=to_read_shelf, book=self.book, user=self.local_user
+ )
+ shelf = self.local_user.shelf_set.get(identifier="reading")
self.assertEqual(to_read_shelf.books.get(), self.book)
self.assertFalse(shelf.books.exists())
self.assertFalse(models.Status.objects.exists())
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.start_reading(request, self.book.id)
self.assertFalse(to_read_shelf.books.exists())
self.assertEqual(shelf.books.get(), self.book)
def test_finish_reading(self):
- ''' begin a book '''
- shelf = self.local_user.shelf_set.get(identifier='read')
+ """ begin a book """
+ shelf = self.local_user.shelf_set.get(identifier="read")
self.assertFalse(shelf.books.exists())
self.assertFalse(models.Status.objects.exists())
readthrough = models.ReadThrough.objects.create(
- user=self.local_user,
- start_date=timezone.now(),
- book=self.book)
+ user=self.local_user, start_date=timezone.now(), book=self.book
+ )
- request = self.factory.post('', {
- 'post-status': True,
- 'privacy': 'followers',
- 'finish_date': '2020-01-07',
- 'id': readthrough.id,
- })
+ request = self.factory.post(
+ "",
+ {
+ "post-status": True,
+ "privacy": "followers",
+ "finish_date": "2020-01-07",
+ "id": readthrough.id,
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.finish_reading(request, self.book.id)
self.assertEqual(shelf.books.get(), self.book)
@@ -107,7 +118,7 @@ class ReadingViews(TestCase):
status = models.GeneratedNote.objects.get()
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.mention_books.get(), self.book)
- self.assertEqual(status.privacy, 'followers')
+ self.assertEqual(status.privacy, "followers")
readthrough = models.ReadThrough.objects.get()
self.assertIsNotNone(readthrough.start_date)
@@ -115,19 +126,21 @@ class ReadingViews(TestCase):
self.assertEqual(readthrough.user, self.local_user)
self.assertEqual(readthrough.book, self.book)
-
def test_edit_readthrough(self):
- ''' adding dates to an ongoing readthrough '''
- start = timezone.make_aware(dateutil.parser.parse('2021-01-03'))
+ """ adding dates to an ongoing readthrough """
+ start = timezone.make_aware(dateutil.parser.parse("2021-01-03"))
readthrough = models.ReadThrough.objects.create(
- book=self.book, user=self.local_user, start_date=start)
+ book=self.book, user=self.local_user, start_date=start
+ )
request = self.factory.post(
- '', {
- 'start_date': '2017-01-01',
- 'finish_date': '2018-03-07',
- 'book': '',
- 'id': readthrough.id,
- })
+ "",
+ {
+ "start_date": "2017-01-01",
+ "finish_date": "2018-03-07",
+ "book": "",
+ "id": readthrough.id,
+ },
+ )
request.user = self.local_user
views.edit_readthrough(request)
@@ -140,33 +153,34 @@ class ReadingViews(TestCase):
self.assertEqual(readthrough.finish_date.day, 7)
self.assertEqual(readthrough.book, self.book)
-
def test_delete_readthrough(self):
- ''' remove a readthrough '''
+ """ remove a readthrough """
readthrough = models.ReadThrough.objects.create(
- book=self.book, user=self.local_user)
- models.ReadThrough.objects.create(
- book=self.book, user=self.local_user)
+ book=self.book, user=self.local_user
+ )
+ models.ReadThrough.objects.create(book=self.book, user=self.local_user)
request = self.factory.post(
- '', {
- 'id': readthrough.id,
- })
+ "",
+ {
+ "id": readthrough.id,
+ },
+ )
request.user = self.local_user
views.delete_readthrough(request)
- self.assertFalse(
- models.ReadThrough.objects.filter(id=readthrough.id).exists())
-
+ self.assertFalse(models.ReadThrough.objects.filter(id=readthrough.id).exists())
def test_create_readthrough(self):
- ''' adding new read dates '''
+ """ adding new read dates """
request = self.factory.post(
- '', {
- 'start_date': '2017-01-01',
- 'finish_date': '2018-03-07',
- 'book': self.book.id,
- 'id': '',
- })
+ "",
+ {
+ "start_date": "2017-01-01",
+ "finish_date": "2018-03-07",
+ "book": self.book.id,
+ "id": "",
+ },
+ )
request.user = self.local_user
views.create_readthrough(request)
diff --git a/bookwyrm/tests/views/test_readthrough.py b/bookwyrm/tests/views/test_readthrough.py
index 41e38a1d..5399f673 100644
--- a/bookwyrm/tests/views/test_readthrough.py
+++ b/bookwyrm/tests/views/test_readthrough.py
@@ -1,4 +1,4 @@
-''' tests updating reading progress '''
+""" tests updating reading progress """
from datetime import datetime
from unittest.mock import patch
from django.test import TestCase, Client
@@ -6,63 +6,68 @@ from django.utils import timezone
from bookwyrm import models
-@patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay')
+
+@patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay")
class ReadThrough(TestCase):
- ''' readthrough tests '''
+ """ readthrough tests """
+
def setUp(self):
- ''' basic user and book data '''
+ """ basic user and book data """
self.client = Client()
- self.work = models.Work.objects.create(
- title='Example Work'
- )
+ self.work = models.Work.objects.create(title="Example Work")
self.edition = models.Edition.objects.create(
- title='Example Edition',
- parent_work=self.work
+ title="Example Edition", parent_work=self.work
)
self.work.default_edition = self.edition
self.work.save()
self.user = models.User.objects.create_user(
- 'cinco', 'cinco@example.com', 'seissiete',
- local=True, localname='cinco')
+ "cinco", "cinco@example.com", "seissiete", local=True, localname="cinco"
+ )
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.client.force_login(self.user)
def test_create_basic_readthrough(self, delay_mock):
"""A basic readthrough doesn't create a progress update"""
self.assertEqual(self.edition.readthrough_set.count(), 0)
- self.client.post('/start-reading/{}'.format(self.edition.id), {
- 'start_date': '2020-11-27',
- })
+ self.client.post(
+ "/start-reading/{}".format(self.edition.id),
+ {
+ "start_date": "2020-11-27",
+ },
+ )
readthroughs = self.edition.readthrough_set.all()
self.assertEqual(len(readthroughs), 1)
self.assertEqual(readthroughs[0].progressupdate_set.count(), 0)
self.assertEqual(
- readthroughs[0].start_date,
- datetime(2020, 11, 27, tzinfo=timezone.utc))
+ readthroughs[0].start_date, datetime(2020, 11, 27, tzinfo=timezone.utc)
+ )
self.assertEqual(readthroughs[0].progress, None)
self.assertEqual(readthroughs[0].finish_date, None)
self.assertEqual(delay_mock.call_count, 1)
def test_create_progress_readthrough(self, delay_mock):
- ''' a readthrough with progress '''
+ """ a readthrough with progress """
self.assertEqual(self.edition.readthrough_set.count(), 0)
- self.client.post('/start-reading/{}'.format(self.edition.id), {
- 'start_date': '2020-11-27',
- 'progress': 50,
- })
+ self.client.post(
+ "/start-reading/{}".format(self.edition.id),
+ {
+ "start_date": "2020-11-27",
+ "progress": 50,
+ },
+ )
readthroughs = self.edition.readthrough_set.all()
self.assertEqual(len(readthroughs), 1)
self.assertEqual(
- readthroughs[0].start_date,
- datetime(2020, 11, 27, tzinfo=timezone.utc))
+ readthroughs[0].start_date, datetime(2020, 11, 27, tzinfo=timezone.utc)
+ )
self.assertEqual(readthroughs[0].progress, 50)
self.assertEqual(readthroughs[0].finish_date, None)
@@ -73,13 +78,17 @@ class ReadThrough(TestCase):
self.assertEqual(delay_mock.call_count, 1)
# Update progress
- self.client.post('/edit-readthrough', {
- 'id': readthroughs[0].id,
- 'progress': 100,
- })
+ self.client.post(
+ "/edit-readthrough",
+ {
+ "id": readthroughs[0].id,
+ "progress": 100,
+ },
+ )
- progress_updates = readthroughs[0].progressupdate_set\
- .order_by('updated_date').all()
+ progress_updates = (
+ readthroughs[0].progressupdate_set.order_by("updated_date").all()
+ )
self.assertEqual(len(progress_updates), 2)
self.assertEqual(progress_updates[1].mode, models.ProgressMode.PAGE)
self.assertEqual(progress_updates[1].progress, 100)
@@ -87,9 +96,12 @@ class ReadThrough(TestCase):
# Edit doesn't publish anything
self.assertEqual(delay_mock.call_count, 1)
- self.client.post('/delete-readthrough', {
- 'id': readthroughs[0].id,
- })
+ self.client.post(
+ "/delete-readthrough",
+ {
+ "id": readthroughs[0].id,
+ },
+ )
readthroughs = self.edition.readthrough_set.all()
updates = self.user.progressupdate_set.all()
diff --git a/bookwyrm/tests/views/test_rss_feed.py b/bookwyrm/tests/views/test_rss_feed.py
index 3d5bec49..c7fef2a9 100644
--- a/bookwyrm/tests/views/test_rss_feed.py
+++ b/bookwyrm/tests/views/test_rss_feed.py
@@ -1,4 +1,4 @@
-''' testing import '''
+""" testing import """
from unittest.mock import patch
from django.test import RequestFactory, TestCase
@@ -6,41 +6,51 @@ from django.test import RequestFactory, TestCase
from bookwyrm import models
from bookwyrm.views import rss_feed
+
class RssFeedView(TestCase):
- ''' rss feed behaves as expected '''
+ """ rss feed behaves as expected """
+
def setUp(self):
- ''' test data '''
+ """ test data """
self.site = models.SiteSettings.objects.create()
self.user = models.User.objects.create_user(
- 'rss_user', 'rss@test.rss', 'password', local=True)
-
- 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
+ "rss_user", "rss@test.rss", "password", local=True
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ 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,
+ )
+
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.review = models.Review.objects.create(
- name='Review name', content='test content', rating=3,
- user=self.user, book=self.book)
+ name="Review name",
+ content="test content",
+ rating=3,
+ user=self.user,
+ book=self.book,
+ )
self.quote = models.Quotation.objects.create(
- quote='a sickening sense', content='test content',
- user=self.user, book=self.book)
+ quote="a sickening sense",
+ content="test content",
+ user=self.user,
+ book=self.book,
+ )
self.generatednote = models.GeneratedNote.objects.create(
- content='test content', user=self.user)
+ content="test content", user=self.user
+ )
self.factory = RequestFactory()
-
def test_rss_feed(self):
- ''' load an rss feed '''
+ """ load an rss feed """
view = rss_feed.RssFeed()
- request = self.factory.get('/user/rss_user/rss')
+ request = self.factory.get("/user/rss_user/rss")
request.user = self.user
with patch("bookwyrm.models.SiteSettings.objects.get") as site:
site.return_value = self.site
diff --git a/bookwyrm/tests/views/test_search.py b/bookwyrm/tests/views/test_search.py
index 655b4563..78c7a103 100644
--- a/bookwyrm/tests/views/test_search.py
+++ b/bookwyrm/tests/views/test_search.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
import json
from unittest.mock import patch
@@ -13,99 +13,107 @@ from bookwyrm.settings import DOMAIN
class ShelfViews(TestCase):
- ''' tag views'''
+ """ tag views"""
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.work = models.Work.objects.create(title='Test Work')
+ self.work = models.Work.objects.create(title="Test Work")
self.book = models.Edition.objects.create(
- title='Test Book',
- remote_id='https://example.com/book/1',
- parent_work=self.work
+ title="Test Book",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
models.Connector.objects.create(
- identifier='self',
- connector_file='self_connector',
- local=True
+ identifier="self", connector_file="self_connector", local=True
)
models.SiteSettings.objects.create()
-
def test_search_json_response(self):
- ''' searches local data only and returns book data in json format '''
+ """ searches local data only and returns book data in json format """
view = views.Search.as_view()
# we need a connector for this, sorry
- request = self.factory.get('', {'q': 'Test Book'})
- with patch('bookwyrm.views.search.is_api_request') as is_api:
+ request = self.factory.get("", {"q": "Test Book"})
+ with patch("bookwyrm.views.search.is_api_request") as is_api:
is_api.return_value = True
response = view(request)
self.assertIsInstance(response, JsonResponse)
data = json.loads(response.content)
self.assertEqual(len(data), 1)
- self.assertEqual(data[0]['title'], 'Test Book')
- self.assertEqual(
- data[0]['key'], 'https://%s/book/%d' % (DOMAIN, self.book.id))
-
+ self.assertEqual(data[0]["title"], "Test Book")
+ self.assertEqual(data[0]["key"], "https://%s/book/%d" % (DOMAIN, self.book.id))
def test_search_html_response(self):
- ''' searches remote connectors '''
+ """ searches remote connectors """
view = views.Search.as_view()
+
class TestConnector(abstract_connector.AbstractMinimalConnector):
- ''' nothing added here '''
+ """ nothing added here """
+
def format_search_result(self, search_result):
pass
+
def get_or_create_book(self, remote_id):
pass
+
def parse_search_data(self, data):
pass
+
+ def format_isbn_search_result(self, search_result):
+ return search_result
+
+ def parse_isbn_search_data(self, data):
+ return data
+
models.Connector.objects.create(
- identifier='example.com',
- connector_file='openlibrary',
- base_url='https://example.com',
- books_url='https://example.com/books',
- covers_url='https://example.com/covers',
- search_url='https://example.com/search?q=',
+ identifier="example.com",
+ connector_file="openlibrary",
+ base_url="https://example.com",
+ books_url="https://example.com/books",
+ covers_url="https://example.com/covers",
+ search_url="https://example.com/search?q=",
)
- connector = TestConnector('example.com')
+ connector = TestConnector("example.com")
search_result = abstract_connector.SearchResult(
- key='http://www.example.com/book/1',
- title='Gideon the Ninth',
- author='Tamsyn Muir',
- year='2019',
- connector=connector
+ key="http://www.example.com/book/1",
+ title="Gideon the Ninth",
+ author="Tamsyn Muir",
+ year="2019",
+ connector=connector,
)
- request = self.factory.get('', {'q': 'Test Book'})
+ request = self.factory.get("", {"q": "Test Book"})
request.user = self.local_user
- with patch('bookwyrm.views.search.is_api_request') as is_api:
+ with patch("bookwyrm.views.search.is_api_request") as is_api:
is_api.return_value = False
- with patch(
- 'bookwyrm.connectors.connector_manager.search') as manager:
+ with patch("bookwyrm.connectors.connector_manager.search") as manager:
manager.return_value = [search_result]
response = view(request)
self.assertIsInstance(response, TemplateResponse)
response.render()
self.assertEqual(
- response.context_data['book_results'][0].title, 'Gideon the Ninth')
-
+ response.context_data["book_results"][0].title, "Gideon the Ninth"
+ )
def test_search_html_response_users(self):
- ''' searches remote connectors '''
+ """ searches remote connectors """
view = views.Search.as_view()
- request = self.factory.get('', {'q': 'mouse'})
+ request = self.factory.get("", {"q": "mouse"})
request.user = self.local_user
- with patch('bookwyrm.views.search.is_api_request') as is_api:
+ with patch("bookwyrm.views.search.is_api_request") as is_api:
is_api.return_value = False
- with patch('bookwyrm.connectors.connector_manager.search'):
+ with patch("bookwyrm.connectors.connector_manager.search"):
response = view(request)
self.assertIsInstance(response, TemplateResponse)
response.render()
- self.assertEqual(
- response.context_data['user_results'][0], self.local_user)
+ self.assertEqual(response.context_data["user_results"][0], self.local_user)
diff --git a/bookwyrm/tests/views/test_shelf.py b/bookwyrm/tests/views/test_shelf.py
index 4fa63689..a308fe56 100644
--- a/bookwyrm/tests/views/test_shelf.py
+++ b/bookwyrm/tests/views/test_shelf.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.template.response import TemplateResponse
from django.test import TestCase
@@ -8,195 +8,172 @@ from bookwyrm import models, views
from bookwyrm.activitypub import ActivitypubResponse
-@patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay')
+@patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay")
class ShelfViews(TestCase):
- ''' tag views'''
+ """ tag views"""
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
self.shelf = models.Shelf.objects.create(
- name='Test Shelf',
- identifier='test-shelf',
- user=self.local_user
+ name="Test Shelf", identifier="test-shelf", user=self.local_user
)
models.SiteSettings.objects.create()
-
def test_shelf_page(self, _):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Shelf.as_view()
shelf = self.local_user.shelf_set.first()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.shelf.is_api_request') as is_api:
+ with patch("bookwyrm.views.shelf.is_api_request") as is_api:
is_api.return_value = False
result = view(request, self.local_user.username, shelf.identifier)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.shelf.is_api_request') as is_api:
+ with patch("bookwyrm.views.shelf.is_api_request") as is_api:
is_api.return_value = True
- result = view(
- request, self.local_user.username, shelf.identifier)
+ result = view(request, self.local_user.username, shelf.identifier)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
- request = self.factory.get('/?page=1')
+ request = self.factory.get("/?page=1")
request.user = self.local_user
- with patch('bookwyrm.views.shelf.is_api_request') as is_api:
+ with patch("bookwyrm.views.shelf.is_api_request") as is_api:
is_api.return_value = True
- result = view(
- request, self.local_user.username, shelf.identifier)
+ result = view(request, self.local_user.username, shelf.identifier)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_edit_shelf_privacy(self, _):
- ''' set name or privacy on shelf '''
+ """ set name or privacy on shelf """
view = views.Shelf.as_view()
- shelf = self.local_user.shelf_set.get(identifier='to-read')
- self.assertEqual(shelf.privacy, 'public')
+ shelf = self.local_user.shelf_set.get(identifier="to-read")
+ self.assertEqual(shelf.privacy, "public")
request = self.factory.post(
- '', {
- 'privacy': 'unlisted',
- 'user': self.local_user.id,
- 'name': 'To Read',
- })
+ "",
+ {
+ "privacy": "unlisted",
+ "user": self.local_user.id,
+ "name": "To Read",
+ },
+ )
request.user = self.local_user
view(request, self.local_user.username, shelf.identifier)
shelf.refresh_from_db()
- self.assertEqual(shelf.privacy, 'unlisted')
-
+ self.assertEqual(shelf.privacy, "unlisted")
def test_edit_shelf_name(self, _):
- ''' change the name of an editable shelf '''
+ """ change the name of an editable shelf """
view = views.Shelf.as_view()
- shelf = models.Shelf.objects.create(
- name='Test Shelf', user=self.local_user)
- self.assertEqual(shelf.privacy, 'public')
+ shelf = models.Shelf.objects.create(name="Test Shelf", user=self.local_user)
+ self.assertEqual(shelf.privacy, "public")
request = self.factory.post(
- '', {
- 'privacy': 'public',
- 'user': self.local_user.id,
- 'name': 'cool name'
- })
+ "", {"privacy": "public", "user": self.local_user.id, "name": "cool name"}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, request.user.username, shelf.identifier)
shelf.refresh_from_db()
- self.assertEqual(shelf.name, 'cool name')
- self.assertEqual(shelf.identifier, 'testshelf-%d' % shelf.id)
-
+ self.assertEqual(shelf.name, "cool name")
+ self.assertEqual(shelf.identifier, "testshelf-%d" % shelf.id)
def test_edit_shelf_name_not_editable(self, _):
- ''' can't change the name of an non-editable shelf '''
+ """ can't change the name of an non-editable shelf """
view = views.Shelf.as_view()
- shelf = self.local_user.shelf_set.get(identifier='to-read')
- self.assertEqual(shelf.privacy, 'public')
+ shelf = self.local_user.shelf_set.get(identifier="to-read")
+ self.assertEqual(shelf.privacy, "public")
request = self.factory.post(
- '', {
- 'privacy': 'public',
- 'user': self.local_user.id,
- 'name': 'cool name'
- })
+ "", {"privacy": "public", "user": self.local_user.id, "name": "cool name"}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request, request.user.username, shelf.identifier)
- self.assertEqual(shelf.name, 'To Read')
-
+ self.assertEqual(shelf.name, "To Read")
def test_handle_shelve(self, _):
- ''' shelve a book '''
- request = self.factory.post('', {
- 'book': self.book.id,
- 'shelf': self.shelf.identifier
- })
+ """ shelve a book """
+ request = self.factory.post(
+ "", {"book": self.book.id, "shelf": self.shelf.identifier}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.shelve(request)
# 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')
- request = self.factory.post('', {
- 'book': self.book.id,
- 'shelf': shelf.identifier
- })
+ """ special behavior for the to-read shelf """
+ shelf = models.Shelf.objects.get(identifier="to-read")
+ request = self.factory.post(
+ "", {"book": self.book.id, "shelf": shelf.identifier}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.shelve(request)
# 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')
- request = self.factory.post('', {
- 'book': self.book.id,
- 'shelf': shelf.identifier
- })
+ """ special behavior for the reading shelf """
+ shelf = models.Shelf.objects.get(identifier="reading")
+ request = self.factory.post(
+ "", {"book": self.book.id, "shelf": shelf.identifier}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.shelve(request)
# 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')
- request = self.factory.post('', {
- 'book': self.book.id,
- 'shelf': shelf.identifier
- })
+ """ special behavior for the read shelf """
+ shelf = models.Shelf.objects.get(identifier="read")
+ request = self.factory.post(
+ "", {"book": self.book.id, "shelf": shelf.identifier}
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.shelve(request)
# 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 '''
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ """ remove a book from a shelf """
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
models.ShelfBook.objects.create(
- book=self.book,
- user=self.local_user,
- shelf=self.shelf
+ book=self.book, user=self.local_user, shelf=self.shelf
)
self.shelf.save()
self.assertEqual(self.shelf.books.count(), 1)
- request = self.factory.post('', {
- 'book': self.book.id,
- 'shelf': self.shelf.id
- })
+ request = self.factory.post("", {"book": self.book.id, "shelf": self.shelf.id})
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
views.unshelve(request)
self.assertEqual(self.shelf.books.count(), 0)
diff --git a/bookwyrm/tests/views/test_status.py b/bookwyrm/tests/views/test_status.py
index 4594a203..7fdc0e06 100644
--- a/bookwyrm/tests/views/test_status.py
+++ b/bookwyrm/tests/views/test_status.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
import json
from unittest.mock import patch
from django.test import TestCase
@@ -9,239 +9,247 @@ from bookwyrm.settings import DOMAIN
class StatusViews(TestCase):
- ''' viewing and creating statuses '''
+ """ viewing and creating statuses """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- with patch('bookwyrm.models.user.set_remote_server'):
+ with patch("bookwyrm.models.user.set_remote_server"):
self.remote_user = models.User.objects.create_user(
- 'rat', 'rat@email.com', 'ratword',
+ "rat",
+ "rat@email.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',
+ remote_id="https://example.com/users/rat",
+ inbox="https://example.com/users/rat/inbox",
+ outbox="https://example.com/users/rat/outbox",
)
- work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=work,
)
-
def test_handle_status(self):
- ''' create a status '''
+ """ create a status """
view = views.CreateStatus.as_view()
- form = forms.CommentForm({
- 'content': 'hi',
- 'user': self.local_user.id,
- 'book': self.book.id,
- 'privacy': 'public',
- })
- request = self.factory.post('', form.data)
+ form = forms.CommentForm(
+ {
+ "content": "hi",
+ "user": self.local_user.id,
+ "book": self.book.id,
+ "privacy": "public",
+ }
+ )
+ request = self.factory.post("", form.data)
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- view(request, 'comment')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ view(request, "comment")
status = models.Comment.objects.get()
- self.assertEqual(status.content, '
hi
')
+ self.assertEqual(status.content, "
hi
")
self.assertEqual(status.user, self.local_user)
self.assertEqual(status.book, self.book)
def test_handle_status_reply(self):
- ''' create a status in reply to an existing status '''
+ """ create a status in reply to an existing status """
view = views.CreateStatus.as_view()
user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'password', local=True)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ "rat", "rat@rat.com", "password", local=True
+ )
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
parent = models.Status.objects.create(
- content='parent status', user=self.local_user)
- form = forms.ReplyForm({
- 'content': 'hi',
- 'user': user.id,
- 'reply_parent': parent.id,
- 'privacy': 'public',
- })
- request = self.factory.post('', form.data)
+ content="parent status", user=self.local_user
+ )
+ form = forms.ReplyForm(
+ {
+ "content": "hi",
+ "user": user.id,
+ "reply_parent": parent.id,
+ "privacy": "public",
+ }
+ )
+ request = self.factory.post("", form.data)
request.user = user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- view(request, 'reply')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ view(request, "reply")
status = models.Status.objects.get(user=user)
- self.assertEqual(status.content, '
hi
')
+ self.assertEqual(status.content, "
hi
")
self.assertEqual(status.user, user)
- self.assertEqual(
- models.Notification.objects.get().user, self.local_user)
+ self.assertEqual(models.Notification.objects.get().user, self.local_user)
def test_handle_status_mentions(self):
- ''' @mention a user in a post '''
+ """ @mention a user in a post """
view = views.CreateStatus.as_view()
user = models.User.objects.create_user(
- 'rat@%s' % DOMAIN, 'rat@rat.com', 'password',
- local=True, localname='rat')
- form = forms.CommentForm({
- 'content': 'hi @rat',
- 'user': self.local_user.id,
- 'book': self.book.id,
- 'privacy': 'public',
- })
- request = self.factory.post('', form.data)
+ "rat@%s" % DOMAIN, "rat@rat.com", "password", local=True, localname="rat"
+ )
+ form = forms.CommentForm(
+ {
+ "content": "hi @rat",
+ "user": self.local_user.id,
+ "book": self.book.id,
+ "privacy": "public",
+ }
+ )
+ request = self.factory.post("", form.data)
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- view(request, 'comment')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ view(request, "comment")
status = models.Status.objects.get()
self.assertEqual(list(status.mention_users.all()), [user])
self.assertEqual(models.Notification.objects.get().user, user)
self.assertEqual(
- status.content,
- '
hi @rat
' % user.remote_id)
+ status.content, '
hi @rat
' % user.remote_id
+ )
def test_handle_status_reply_with_mentions(self):
- ''' reply to a post with an @mention'ed user '''
+ """ reply to a post with an @mention'ed user """
view = views.CreateStatus.as_view()
user = models.User.objects.create_user(
- 'rat', 'rat@rat.com', 'password',
- local=True, localname='rat')
- form = forms.CommentForm({
- 'content': 'hi @rat@example.com',
- 'user': self.local_user.id,
- 'book': self.book.id,
- 'privacy': 'public',
- })
- request = self.factory.post('', form.data)
+ "rat", "rat@rat.com", "password", local=True, localname="rat"
+ )
+ form = forms.CommentForm(
+ {
+ "content": "hi @rat@example.com",
+ "user": self.local_user.id,
+ "book": self.book.id,
+ "privacy": "public",
+ }
+ )
+ request = self.factory.post("", form.data)
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- view(request, 'comment')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ view(request, "comment")
status = models.Status.objects.get()
- form = forms.ReplyForm({
- 'content': 'right',
- 'user': user.id,
- 'privacy': 'public',
- 'reply_parent': status.id
- })
- request = self.factory.post('', form.data)
+ form = forms.ReplyForm(
+ {
+ "content": "right",
+ "user": user.id,
+ "privacy": "public",
+ "reply_parent": status.id,
+ }
+ )
+ request = self.factory.post("", form.data)
request.user = user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- view(request, 'reply')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ view(request, "reply")
reply = models.Status.replies(status).first()
- self.assertEqual(reply.content, '
right
')
+ self.assertEqual(reply.content, "
right
")
self.assertEqual(reply.user, user)
# the mentioned user in the parent post is only included if @'ed
self.assertFalse(self.remote_user in reply.mention_users.all())
self.assertTrue(self.local_user in reply.mention_users.all())
def test_find_mentions(self):
- ''' detect and look up @ mentions of users '''
+ """ detect and look up @ mentions of users """
user = models.User.objects.create_user(
- 'nutria@%s' % DOMAIN, 'nutria@nutria.com', 'password',
- local=True, localname='nutria')
- self.assertEqual(user.username, 'nutria@%s' % DOMAIN)
+ "nutria@%s" % DOMAIN,
+ "nutria@nutria.com",
+ "password",
+ local=True,
+ localname="nutria",
+ )
+ self.assertEqual(user.username, "nutria@%s" % DOMAIN)
self.assertEqual(
- list(views.status.find_mentions('@nutria'))[0],
- ('@nutria', user)
+ list(views.status.find_mentions("@nutria"))[0], ("@nutria", user)
)
self.assertEqual(
- list(views.status.find_mentions('leading text @nutria'))[0],
- ('@nutria', user)
+ list(views.status.find_mentions("leading text @nutria"))[0],
+ ("@nutria", user),
)
self.assertEqual(
- list(views.status.find_mentions(
- 'leading @nutria trailing text'))[0],
- ('@nutria', user)
+ list(views.status.find_mentions("leading @nutria trailing text"))[0],
+ ("@nutria", user),
)
self.assertEqual(
- list(views.status.find_mentions(
- '@rat@example.com'))[0],
- ('@rat@example.com', self.remote_user)
+ list(views.status.find_mentions("@rat@example.com"))[0],
+ ("@rat@example.com", self.remote_user),
)
- multiple = list(views.status.find_mentions(
- '@nutria and @rat@example.com'))
- self.assertEqual(multiple[0], ('@nutria', user))
- self.assertEqual(multiple[1], ('@rat@example.com', self.remote_user))
+ multiple = list(views.status.find_mentions("@nutria and @rat@example.com"))
+ self.assertEqual(multiple[0], ("@nutria", user))
+ self.assertEqual(multiple[1], ("@rat@example.com", self.remote_user))
- with patch('bookwyrm.views.status.handle_remote_webfinger') as rw:
+ with patch("bookwyrm.views.status.handle_remote_webfinger") as rw:
rw.return_value = self.local_user
self.assertEqual(
- list(views.status.find_mentions('@beep@beep.com'))[0],
- ('@beep@beep.com', self.local_user)
+ list(views.status.find_mentions("@beep@beep.com"))[0],
+ ("@beep@beep.com", self.local_user),
)
- with patch('bookwyrm.views.status.handle_remote_webfinger') as rw:
+ with patch("bookwyrm.views.status.handle_remote_webfinger") as rw:
rw.return_value = None
- self.assertEqual(list(views.status.find_mentions(
- '@beep@beep.com')), [])
+ self.assertEqual(list(views.status.find_mentions("@beep@beep.com")), [])
self.assertEqual(
- list(views.status.find_mentions('@nutria@%s' % DOMAIN))[0],
- ('@nutria@%s' % DOMAIN, user)
+ list(views.status.find_mentions("@nutria@%s" % DOMAIN))[0],
+ ("@nutria@%s" % DOMAIN, user),
)
def test_format_links(self):
- ''' find and format urls into a tags '''
- url = 'http://www.fish.com/'
+ """ find and format urls into a tags """
+ url = "http://www.fish.com/"
+ self.assertEqual(
+ views.status.format_links(url), '
www.fish.com/ ' % url
+ )
+ self.assertEqual(
+ views.status.format_links("(%s)" % url),
+ '(
www.fish.com/ )' % url,
+ )
+ url = "https://archive.org/details/dli.granth.72113/page/n25/mode/2up"
self.assertEqual(
views.status.format_links(url),
- '
www.fish.com/ ' % url)
- self.assertEqual(
- views.status.format_links('(%s)' % url),
- '(
www.fish.com/ )' % url)
- url = 'https://archive.org/details/dli.granth.72113/page/n25/mode/2up'
+ '
'
+ "archive.org/details/dli.granth.72113/page/n25/mode/2up " % url,
+ )
+ url = "https://openlibrary.org/search" "?q=arkady+strugatsky&mode=everything"
self.assertEqual(
views.status.format_links(url),
- '
' \
- 'archive.org/details/dli.granth.72113/page/n25/mode/2up ' \
- % url)
- url = 'https://openlibrary.org/search' \
- '?q=arkady+strugatsky&mode=everything'
- self.assertEqual(
- views.status.format_links(url),
- '
openlibrary.org/search' \
- '?q=arkady+strugatsky&mode=everything ' % url)
-
+ '
openlibrary.org/search'
+ "?q=arkady+strugatsky&mode=everything " % url,
+ )
def test_to_markdown(self):
- ''' this is mostly handled in other places, but nonetheless '''
- text = '_hi_ and http://fish.com is
rad '
+ """ this is mostly handled in other places, but nonetheless """
+ text = "_hi_ and http://fish.com is
rad "
result = views.status.to_markdown(text)
self.assertEqual(
result,
- '
hi and fish.com ' \
- 'is rad
')
-
+ '
hi and fish.com ' "is rad
",
+ )
def test_to_markdown_link(self):
- ''' this is mostly handled in other places, but nonetheless '''
- text = '[hi](http://fish.com) is
rad '
+ """ this is mostly handled in other places, but nonetheless """
+ text = "[hi](http://fish.com) is
rad "
result = views.status.to_markdown(text)
- self.assertEqual(
- result,
- '
hi ' \
- 'is rad
')
-
+ self.assertEqual(result, '
hi ' "is rad
")
def test_handle_delete_status(self):
- ''' marks a status as deleted '''
+ """ marks a status as deleted """
view = views.DeleteStatus.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- status = models.Status.objects.create(
- user=self.local_user, content='hi')
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ status = models.Status.objects.create(user=self.local_user, content="hi")
self.assertFalse(status.deleted)
- request = self.factory.post('')
+ request = self.factory.post("")
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
- as mock:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay") as mock:
view(request, status.id)
activity = json.loads(mock.call_args_list[0][0][1])
- self.assertEqual(activity['type'], 'Delete')
- self.assertEqual(activity['object']['type'], 'Tombstone')
+ self.assertEqual(activity["type"], "Delete")
+ self.assertEqual(activity["object"]["type"], "Tombstone")
status.refresh_from_db()
self.assertTrue(status.deleted)
diff --git a/bookwyrm/tests/views/test_tag.py b/bookwyrm/tests/views/test_tag.py
index ef809b46..6ad6ab25 100644
--- a/bookwyrm/tests/views/test_tag.py
+++ b/bookwyrm/tests/views/test_tag.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
from unittest.mock import patch
from django.contrib.auth.models import Group, Permission
from django.contrib.contenttypes.models import ContentType
@@ -11,107 +11,109 @@ from bookwyrm.activitypub import ActivitypubResponse
class TagViews(TestCase):
- ''' tag views'''
+ """ tag views"""
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
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',
+ "mouse@local.com",
+ "mouse@mouse.com",
+ "mouseword",
+ local=True,
+ localname="mouse",
+ remote_id="https://example.com/users/mouse",
)
- self.group = Group.objects.create(name='editor')
+ self.group = Group.objects.create(name="editor")
self.group.permissions.add(
Permission.objects.create(
- name='edit_book',
- codename='edit_book',
- content_type=ContentType.objects.get_for_model(models.User)).id
+ name="edit_book",
+ codename="edit_book",
+ content_type=ContentType.objects.get_for_model(models.User),
+ ).id
)
- self.work = models.Work.objects.create(title='Test Work')
+ 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
+ title="Example Edition",
+ remote_id="https://example.com/book/1",
+ parent_work=self.work,
)
models.SiteSettings.objects.create()
-
def test_tag_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Tag.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- tag = models.Tag.objects.create(name='hi there')
- models.UserTag.objects.create(
- tag=tag, user=self.local_user, book=self.book)
- request = self.factory.get('')
- with patch('bookwyrm.views.tag.is_api_request') as is_api:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ tag = models.Tag.objects.create(name="hi there")
+ models.UserTag.objects.create(tag=tag, user=self.local_user, book=self.book)
+ request = self.factory.get("")
+ with patch("bookwyrm.views.tag.is_api_request") as is_api:
is_api.return_value = False
result = view(request, tag.identifier)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- request = self.factory.get('')
- with patch('bookwyrm.views.tag.is_api_request') as is_api:
+ request = self.factory.get("")
+ with patch("bookwyrm.views.tag.is_api_request") as is_api:
is_api.return_value = True
result = view(request, tag.identifier)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_tag_page_activitypub_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Tag.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- tag = models.Tag.objects.create(name='hi there')
- models.UserTag.objects.create(
- tag=tag, user=self.local_user, book=self.book)
- request = self.factory.get('', {'page': 1})
- with patch('bookwyrm.views.tag.is_api_request') as is_api:
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ tag = models.Tag.objects.create(name="hi there")
+ models.UserTag.objects.create(tag=tag, user=self.local_user, book=self.book)
+ request = self.factory.get("", {"page": 1})
+ with patch("bookwyrm.views.tag.is_api_request") as is_api:
is_api.return_value = True
result = view(request, tag.identifier)
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_tag(self):
- ''' add a tag to a book '''
+ """ add a tag to a book """
view = views.AddTag.as_view()
request = self.factory.post(
- '', {
- 'name': 'A Tag!?',
- 'book': self.book.id,
- })
+ "",
+ {
+ "name": "A Tag!?",
+ "book": self.book.id,
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request)
tag = models.Tag.objects.get()
user_tag = models.UserTag.objects.get()
- self.assertEqual(tag.name, 'A Tag!?')
- self.assertEqual(tag.identifier, 'A+Tag%21%3F')
+ self.assertEqual(tag.name, "A Tag!?")
+ self.assertEqual(tag.identifier, "A+Tag%21%3F")
self.assertEqual(user_tag.user, self.local_user)
self.assertEqual(user_tag.book, self.book)
-
def test_untag(self):
- ''' remove a tag from a book '''
+ """ remove a tag from a book """
view = views.RemoveTag.as_view()
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
- tag = models.Tag.objects.create(name='A Tag!?')
- models.UserTag.objects.create(
- user=self.local_user, book=self.book, tag=tag)
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
+ tag = models.Tag.objects.create(name="A Tag!?")
+ models.UserTag.objects.create(user=self.local_user, book=self.book, tag=tag)
request = self.factory.post(
- '', {
- 'user': self.local_user.id,
- 'book': self.book.id,
- 'name': tag.name,
- })
+ "",
+ {
+ "user": self.local_user.id,
+ "book": self.book.id,
+ "name": tag.name,
+ },
+ )
request.user = self.local_user
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay'):
+ with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
view(request)
- self.assertTrue(models.Tag.objects.filter(name='A Tag!?').exists())
+ self.assertTrue(models.Tag.objects.filter(name="A Tag!?").exists())
self.assertFalse(models.UserTag.objects.exists())
diff --git a/bookwyrm/tests/views/test_user.py b/bookwyrm/tests/views/test_user.py
index f08c17e7..596ea8bf 100644
--- a/bookwyrm/tests/views/test_user.py
+++ b/bookwyrm/tests/views/test_user.py
@@ -1,4 +1,4 @@
-''' test for app action functionality '''
+""" test for app action functionality """
import pathlib
from unittest.mock import patch
from PIL import Image
@@ -15,180 +15,177 @@ from bookwyrm.activitypub import ActivitypubResponse
class UserViews(TestCase):
- ''' view user and edit profile '''
+ """ view user and edit profile """
+
def setUp(self):
- ''' we need basic test data and mocks '''
+ """ we need basic test data and mocks """
self.factory = RequestFactory()
self.local_user = models.User.objects.create_user(
- 'mouse@local.com', 'mouse@mouse.mouse', 'password',
- local=True, localname='mouse')
+ "mouse@local.com",
+ "mouse@mouse.mouse",
+ "password",
+ local=True,
+ localname="mouse",
+ )
self.rat = models.User.objects.create_user(
- 'rat@local.com', 'rat@rat.rat', 'password',
- local=True, localname='rat')
+ "rat@local.com", "rat@rat.rat", "password", local=True, localname="rat"
+ )
models.SiteSettings.objects.create()
self.anonymous_user = AnonymousUser
self.anonymous_user.is_authenticated = False
-
def test_user_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.User.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
request.user = self.anonymous_user
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = True
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_user_page_blocked(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.User.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
self.rat.blocks.add(self.local_user)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'rat')
+ result = view(request, "rat")
self.assertEqual(result.status_code, 404)
-
def test_followers_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Followers.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = True
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_followers_page_blocked(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Followers.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
self.rat.blocks.add(self.local_user)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'rat')
+ result = view(request, "rat")
self.assertEqual(result.status_code, 404)
-
def test_following_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Following.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = True
- result = view(request, 'mouse')
+ result = view(request, "mouse")
self.assertIsInstance(result, ActivitypubResponse)
self.assertEqual(result.status_code, 200)
-
def test_following_page_blocked(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.Following.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
self.rat.blocks.add(self.local_user)
- with patch('bookwyrm.views.user.is_api_request') as is_api:
+ with patch("bookwyrm.views.user.is_api_request") as is_api:
is_api.return_value = False
- result = view(request, 'rat')
+ result = view(request, "rat")
self.assertEqual(result.status_code, 404)
-
def test_edit_user_page(self):
- ''' there are so many views, this just makes sure it LOADS '''
+ """ there are so many views, this just makes sure it LOADS """
view = views.EditUser.as_view()
- request = self.factory.get('')
+ request = self.factory.get("")
request.user = self.local_user
result = view(request)
self.assertIsInstance(result, TemplateResponse)
result.render()
self.assertEqual(result.status_code, 200)
-
def test_edit_user(self):
- ''' use a form to update a user '''
+ """ use a form to update a user """
view = views.EditUser.as_view()
form = forms.EditUserForm(instance=self.local_user)
- form.data['name'] = 'New Name'
- form.data['email'] = 'wow@email.com'
- request = self.factory.post('', form.data)
+ form.data["name"] = "New Name"
+ form.data["email"] = "wow@email.com"
+ request = self.factory.post("", form.data)
request.user = self.local_user
self.assertIsNone(self.local_user.name)
- with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
- as delay_mock:
+ with patch(
+ "bookwyrm.models.activitypub_mixin.broadcast_task.delay"
+ ) as delay_mock:
view(request)
self.assertEqual(delay_mock.call_count, 1)
- self.assertEqual(self.local_user.name, 'New Name')
- self.assertEqual(self.local_user.email, 'wow@email.com')
+ self.assertEqual(self.local_user.name, "New Name")
+ self.assertEqual(self.local_user.email, "wow@email.com")
+ # idk how to mock the upload form, got tired of triyng to make it work
+ # def test_edit_user_avatar(self):
+ # ''' use a form to update a user '''
+ # view = views.EditUser.as_view()
+ # form = forms.EditUserForm(instance=self.local_user)
+ # form.data['name'] = 'New Name'
+ # form.data['email'] = 'wow@email.com'
+ # image_file = pathlib.Path(__file__).parent.joinpath(
+ # '../../static/images/no_cover.jpg')
+ # image = Image.open(image_file)
+ # form.files['avatar'] = SimpleUploadedFile(
+ # image_file, open(image_file), content_type='image/jpeg')
+ # request = self.factory.post('', form.data, form.files)
+ # request.user = self.local_user
-# idk how to mock the upload form, got tired of triyng to make it work
-# def test_edit_user_avatar(self):
-# ''' use a form to update a user '''
-# view = views.EditUser.as_view()
-# form = forms.EditUserForm(instance=self.local_user)
-# form.data['name'] = 'New Name'
-# form.data['email'] = 'wow@email.com'
-# image_file = pathlib.Path(__file__).parent.joinpath(
-# '../../static/images/no_cover.jpg')
-# image = Image.open(image_file)
-# form.files['avatar'] = SimpleUploadedFile(
-# image_file, open(image_file), content_type='image/jpeg')
-# request = self.factory.post('', form.data, form.files)
-# request.user = self.local_user
-
-# with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
-# as delay_mock:
-# view(request)
-# self.assertEqual(delay_mock.call_count, 1)
-# self.assertEqual(self.local_user.name, 'New Name')
-# self.assertEqual(self.local_user.email, 'wow@email.com')
-# self.assertIsNotNone(self.local_user.avatar)
-# self.assertEqual(self.local_user.avatar.size, (120, 120))
-
+ # with patch('bookwyrm.models.activitypub_mixin.broadcast_task.delay') \
+ # as delay_mock:
+ # view(request)
+ # self.assertEqual(delay_mock.call_count, 1)
+ # self.assertEqual(self.local_user.name, 'New Name')
+ # self.assertEqual(self.local_user.email, 'wow@email.com')
+ # self.assertIsNotNone(self.local_user.avatar)
+ # self.assertEqual(self.local_user.avatar.size, (120, 120))
def test_crop_avatar(self):
- ''' reduce that image size '''
+ """ reduce that image size """
image_file = pathlib.Path(__file__).parent.joinpath(
- '../../static/images/no_cover.jpg')
+ "../../static/images/no_cover.jpg"
+ )
image = Image.open(image_file)
result = views.user.crop_avatar(image)
diff --git a/bookwyrm/urls.py b/bookwyrm/urls.py
index 76054e3f..844f8993 100644
--- a/bookwyrm/urls.py
+++ b/bookwyrm/urls.py
@@ -1,4 +1,4 @@
-''' url routing for the app and api '''
+""" url routing for the app and api """
from django.conf.urls.static import static
from django.contrib import admin
from django.urls import path, re_path
@@ -7,168 +7,165 @@ from django.urls import path, re_path
from bookwyrm import settings, views, wellknown
from bookwyrm.utils import regex
-user_path = r'^user/(?P
%s)' % regex.username
-local_user_path = r'^user/(?P%s)' % regex.localname
+user_path = r"^user/(?P%s)" % regex.username
+local_user_path = r"^user/(?P%s)" % regex.localname
status_types = [
- 'status',
- 'review',
- 'reviewrating',
- 'comment',
- 'quotation',
- 'boost',
- 'generatednote'
+ "status",
+ "review",
+ "reviewrating",
+ "comment",
+ "quotation",
+ "boost",
+ "generatednote",
]
-status_path = r'%s/(%s)/(?P\d+)' % \
- (user_path, '|'.join(status_types))
+status_path = r"%s/(%s)/(?P\d+)" % (user_path, "|".join(status_types))
-book_path = r'^book/(?P\d+)'
+book_path = r"^book/(?P\d+)"
-handler404 = 'bookwyrm.views.not_found_page'
-handler500 = 'bookwyrm.views.server_error_page'
+handler404 = "bookwyrm.views.not_found_page"
+handler500 = "bookwyrm.views.server_error_page"
urlpatterns = [
- path('admin/', admin.site.urls),
-
+ path("admin/", admin.site.urls),
# federation endpoints
- re_path(r'^inbox/?$', views.Inbox.as_view()),
- re_path(r'%s/inbox/?$' % local_user_path, views.Inbox.as_view()),
- re_path(r'%s/outbox/?$' % local_user_path, views.Outbox.as_view()),
- re_path(r'^.well-known/webfinger/?$', wellknown.webfinger),
- re_path(r'^.well-known/nodeinfo/?$', wellknown.nodeinfo_pointer),
- re_path(r'^nodeinfo/2\.0/?$', wellknown.nodeinfo),
- re_path(r'^api/v1/instance/?$', wellknown.instance_info),
- re_path(r'^api/v1/instance/peers/?$', wellknown.peers),
-
+ re_path(r"^inbox/?$", views.Inbox.as_view()),
+ re_path(r"%s/inbox/?$" % local_user_path, views.Inbox.as_view()),
+ re_path(r"%s/outbox/?$" % local_user_path, views.Outbox.as_view()),
+ re_path(r"^.well-known/webfinger/?$", wellknown.webfinger),
+ re_path(r"^.well-known/nodeinfo/?$", wellknown.nodeinfo_pointer),
+ re_path(r"^nodeinfo/2\.0/?$", wellknown.nodeinfo),
+ re_path(r"^api/v1/instance/?$", wellknown.instance_info),
+ re_path(r"^api/v1/instance/peers/?$", wellknown.peers),
# polling updates
- re_path('^api/updates/notifications/?$', views.Updates.as_view()),
-
+ re_path("^api/updates/notifications/?$", views.Updates.as_view()),
# authentication
- re_path(r'^login/?$', views.Login.as_view()),
- re_path(r'^register/?$', views.Register.as_view()),
- re_path(r'^logout/?$', views.Logout.as_view()),
- re_path(r'^password-reset/?$', views.PasswordResetRequest.as_view()),
- re_path(r'^password-reset/(?P[A-Za-z0-9]+)/?$',
- views.PasswordReset.as_view()),
-
+ re_path(r"^login/?$", views.Login.as_view()),
+ re_path(r"^register/?$", views.Register.as_view()),
+ re_path(r"^logout/?$", views.Logout.as_view()),
+ re_path(r"^password-reset/?$", views.PasswordResetRequest.as_view()),
+ re_path(
+ r"^password-reset/(?P[A-Za-z0-9]+)/?$", views.PasswordReset.as_view()
+ ),
# admin
- re_path(r'^settings/site-settings',
- views.Site.as_view(), name='settings-site'),
- re_path(r'^settings/federation',
- views.Federation.as_view(), name='settings-federation'),
- re_path(r'^settings/invites/?$',
- views.ManageInvites.as_view(), name='settings-invites'),
- re_path(r'^invite/(?P[A-Za-z0-9]+)/?$', views.Invite.as_view()),
-
+ re_path(r"^settings/site-settings", views.Site.as_view(), name="settings-site"),
+ re_path(
+ r"^settings/federation", views.Federation.as_view(), name="settings-federation"
+ ),
+ re_path(
+ r"^settings/invites/?$", views.ManageInvites.as_view(), name="settings-invites"
+ ),
+ re_path(r"^invite/(?P[A-Za-z0-9]+)/?$", views.Invite.as_view()),
# landing pages
- re_path(r'^about/?$', views.About.as_view()),
- path('', views.Home.as_view()),
- re_path(r'^discover/?$', views.Discover.as_view()),
- re_path(r'^notifications/?$', views.Notifications.as_view()),
-
+ re_path(r"^about/?$", views.About.as_view()),
+ path("", views.Home.as_view()),
+ re_path(r"^discover/?$", views.Discover.as_view()),
+ re_path(r"^notifications/?$", views.Notifications.as_view()),
# feeds
- re_path(r'^(?Phome|local|federated)/?$', views.Feed.as_view()),
- re_path(r'^direct-messages/?$', views.DirectMessage.as_view()),
- re_path(r'^direct-messages/(?P%s)?$' % regex.username,
- views.DirectMessage.as_view()),
-
+ re_path(r"^(?Phome|local|federated)/?$", views.Feed.as_view()),
+ re_path(r"^direct-messages/?$", views.DirectMessage.as_view()),
+ re_path(
+ r"^direct-messages/(?P%s)?$" % regex.username,
+ views.DirectMessage.as_view(),
+ ),
# search
- re_path(r'^search/?$', views.Search.as_view()),
-
+ re_path(r"^search/?$", views.Search.as_view()),
# imports
- re_path(r'^import/?$', views.Import.as_view()),
- re_path(r'^import/(\d+)/?$', views.ImportStatus.as_view()),
-
+ re_path(r"^import/?$", views.Import.as_view()),
+ re_path(r"^import/(\d+)/?$", views.ImportStatus.as_view()),
# users
- re_path(r'%s/?$' % user_path, views.User.as_view(), name='user-feed'),
- re_path(r'%s\.json$' % user_path, views.User.as_view()),
- re_path(r'%s/rss' % user_path, views.rss_feed.RssFeed(), name='user-rss'),
- re_path(r'%s/followers(.json)?/?$' % user_path,
- views.Followers.as_view(), name='user-followers'),
- re_path(r'%s/following(.json)?/?$' % user_path,
- views.Following.as_view(), name='user-following'),
- re_path(r'%s/shelves/?$' % user_path,
- views.user_shelves_page, name='user-shelves'),
- re_path(r'%s/lists/?$' % user_path,
- views.UserLists.as_view(), name='user-lists'),
- re_path(r'%s/goal/(?P\d{4})/?$' % user_path,
- views.Goal.as_view(), name='user-goal'),
-
-
+ re_path(r"%s/?$" % user_path, views.User.as_view(), name="user-feed"),
+ re_path(r"%s\.json$" % user_path, views.User.as_view()),
+ re_path(r"%s/rss" % user_path, views.rss_feed.RssFeed(), name="user-rss"),
+ re_path(
+ r"%s/followers(.json)?/?$" % user_path,
+ views.Followers.as_view(),
+ name="user-followers",
+ ),
+ re_path(
+ r"%s/following(.json)?/?$" % user_path,
+ views.Following.as_view(),
+ name="user-following",
+ ),
+ re_path(r"%s/shelves/?$" % user_path, views.user_shelves_page, name="user-shelves"),
+ re_path(r"%s/lists/?$" % user_path, views.UserLists.as_view(), name="user-lists"),
+ re_path(
+ r"%s/goal/(?P\d{4})/?$" % user_path,
+ views.Goal.as_view(),
+ name="user-goal",
+ ),
# lists
- re_path(r'^list/?$', views.Lists.as_view(), name='lists'),
- re_path(r'^list/(?P\d+)(.json)?/?$',
- views.List.as_view(), name='list'),
- re_path(r'^list/(?P\d+)/add/?$',
- views.list.add_book, name='list-add-book'),
- re_path(r'^list/(?P\d+)/remove/?$',
- views.list.remove_book, name='list-remove-book'),
- re_path(r'^list/(?P\d+)/curate/?$',
- views.Curate.as_view(), name='list-curate'),
-
+ re_path(r"^list/?$", views.Lists.as_view(), name="lists"),
+ re_path(r"^list/(?P\d+)(.json)?/?$", views.List.as_view(), name="list"),
+ re_path(
+ r"^list/(?P\d+)/add/?$", views.list.add_book, name="list-add-book"
+ ),
+ re_path(
+ r"^list/(?P\d+)/remove/?$",
+ views.list.remove_book,
+ name="list-remove-book",
+ ),
+ re_path(
+ r"^list/(?P\d+)/curate/?$", views.Curate.as_view(), name="list-curate"
+ ),
# preferences
- re_path(r'^preferences/profile/?$',
- views.EditUser.as_view(), name='prefs-profile'),
- re_path(r'^preferences/password/?$', views.ChangePassword.as_view()),
- re_path(r'^preferences/block/?$', views.Block.as_view()),
- re_path(r'^block/(?P\d+)/?$', views.Block.as_view()),
- re_path(r'^unblock/(?P\d+)/?$', views.unblock),
-
+ re_path(r"^preferences/profile/?$", views.EditUser.as_view(), name="prefs-profile"),
+ re_path(r"^preferences/password/?$", views.ChangePassword.as_view()),
+ re_path(r"^preferences/block/?$", views.Block.as_view()),
+ re_path(r"^block/(?P\d+)/?$", views.Block.as_view()),
+ re_path(r"^unblock/(?P\d+)/?$", views.unblock),
# statuses
- re_path(r'%s(.json)?/?$' % status_path, views.Status.as_view()),
- re_path(r'%s/activity/?$' % status_path, views.Status.as_view()),
- re_path(r'%s/replies(.json)?/?$' % status_path, views.Replies.as_view()),
- re_path(r'^post/(?P\w+)/?$', views.CreateStatus.as_view()),
- re_path(r'^delete-status/(?P\d+)/?$',
- views.DeleteStatus.as_view()),
-
+ re_path(r"%s(.json)?/?$" % status_path, views.Status.as_view()),
+ re_path(r"%s/activity/?$" % status_path, views.Status.as_view()),
+ re_path(r"%s/replies(.json)?/?$" % status_path, views.Replies.as_view()),
+ re_path(r"^post/(?P\w+)/?$", views.CreateStatus.as_view()),
+ re_path(r"^delete-status/(?P\d+)/?$", views.DeleteStatus.as_view()),
# interact
- re_path(r'^favorite/(?P\d+)/?$', views.Favorite.as_view()),
- re_path(r'^unfavorite/(?P\d+)/?$', views.Unfavorite.as_view()),
- re_path(r'^boost/(?P\d+)/?$', views.Boost.as_view()),
- re_path(r'^unboost/(?P\d+)/?$', views.Unboost.as_view()),
-
+ re_path(r"^favorite/(?P\d+)/?$", views.Favorite.as_view()),
+ re_path(r"^unfavorite/(?P\d+)/?$", views.Unfavorite.as_view()),
+ re_path(r"^boost/(?P\d+)/?$", views.Boost.as_view()),
+ re_path(r"^unboost/(?P\d+)/?$", views.Unboost.as_view()),
# books
- re_path(r'%s(.json)?/?$' % book_path, views.Book.as_view()),
- re_path(r'%s/edit/?$' % book_path, views.EditBook.as_view()),
- re_path(r'%s/editions(.json)?/?$' % book_path, views.Editions.as_view()),
- re_path(r'^upload-cover/(?P\d+)/?$', views.upload_cover),
- re_path(r'^add-description/(?P\d+)/?$', views.add_description),
- re_path(r'^resolve-book/?$', views.resolve_book),
- re_path(r'^switch-edition/?$', views.switch_edition),
-
+ re_path(r"%s(.json)?/?$" % book_path, views.Book.as_view()),
+ re_path(r"%s/edit/?$" % book_path, views.EditBook.as_view()),
+ re_path(r"%s/editions(.json)?/?$" % book_path, views.Editions.as_view()),
+ re_path(r"^upload-cover/(?P\d+)/?$", views.upload_cover),
+ re_path(r"^add-description/(?P\d+)/?$", views.add_description),
+ re_path(r"^resolve-book/?$", views.resolve_book),
+ re_path(r"^switch-edition/?$", views.switch_edition),
+ # isbn
+ re_path(r"^isbn/(?P\d+)(.json)?/?$", views.Isbn.as_view()),
# author
- re_path(r'^author/(?P\d+)(.json)?/?$', views.Author.as_view()),
- re_path(r'^author/(?P\d+)/edit/?$', views.EditAuthor.as_view()),
-
+ re_path(r"^author/(?P\d+)(.json)?/?$", views.Author.as_view()),
+ re_path(r"^author/(?P\d+)/edit/?$", views.EditAuthor.as_view()),
# tags
- re_path(r'^tag/(?P.+)\.json/?$', views.Tag.as_view()),
- re_path(r'^tag/(?P.+)/?$', views.Tag.as_view()),
- re_path(r'^tag/?$', views.AddTag.as_view()),
- re_path(r'^untag/?$', views.RemoveTag.as_view()),
-
+ re_path(r"^tag/(?P.+)\.json/?$", views.Tag.as_view()),
+ re_path(r"^tag/(?P.+)/?$", views.Tag.as_view()),
+ re_path(r"^tag/?$", views.AddTag.as_view()),
+ re_path(r"^untag/?$", views.RemoveTag.as_view()),
# shelf
- re_path(r'^%s/shelf/(?P[\w-]+)(.json)?/?$' % \
- user_path, views.Shelf.as_view(), name='shelf'),
- re_path(r'^%s/shelf/(?P[\w-]+)(.json)?/?$' % \
- local_user_path, views.Shelf.as_view()),
- re_path(r'^create-shelf/?$', views.create_shelf, name='shelf-create'),
- re_path(r'^delete-shelf/(?P\d+)?$', views.delete_shelf),
- re_path(r'^shelve/?$', views.shelve),
- re_path(r'^unshelve/?$', views.unshelve),
-
+ re_path(
+ r"^%s/shelf/(?P[\w-]+)(.json)?/?$" % user_path,
+ views.Shelf.as_view(),
+ name="shelf",
+ ),
+ re_path(
+ r"^%s/shelf/(?P[\w-]+)(.json)?/?$" % local_user_path,
+ views.Shelf.as_view(),
+ ),
+ re_path(r"^create-shelf/?$", views.create_shelf, name="shelf-create"),
+ re_path(r"^delete-shelf/(?P\d+)?$", views.delete_shelf),
+ re_path(r"^shelve/?$", views.shelve),
+ re_path(r"^unshelve/?$", views.unshelve),
# reading progress
- re_path(r'^edit-readthrough/?$', views.edit_readthrough),
- re_path(r'^delete-readthrough/?$', views.delete_readthrough),
- re_path(r'^create-readthrough/?$', views.create_readthrough),
- re_path(r'^delete-progressupdate/?$', views.delete_progressupdate),
-
- re_path(r'^start-reading/(?P\d+)/?$', views.start_reading),
- re_path(r'^finish-reading/(?P\d+)/?$', views.finish_reading),
-
+ re_path(r"^edit-readthrough/?$", views.edit_readthrough),
+ re_path(r"^delete-readthrough/?$", views.delete_readthrough),
+ re_path(r"^create-readthrough/?$", views.create_readthrough),
+ re_path(r"^delete-progressupdate/?$", views.delete_progressupdate),
+ re_path(r"^start-reading/(?P\d+)/?$", views.start_reading),
+ re_path(r"^finish-reading/(?P\d+)/?$", views.finish_reading),
# following
- re_path(r'^follow/?$', views.follow),
- re_path(r'^unfollow/?$', views.unfollow),
- re_path(r'^accept-follow-request/?$', views.accept_follow_request),
- re_path(r'^delete-follow-request/?$', views.delete_follow_request),
-
+ re_path(r"^follow/?$", views.follow),
+ re_path(r"^unfollow/?$", views.unfollow),
+ re_path(r"^accept-follow-request/?$", views.accept_follow_request),
+ re_path(r"^delete-follow-request/?$", views.delete_follow_request),
] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
diff --git a/bookwyrm/utils/regex.py b/bookwyrm/utils/regex.py
index c818bc41..6389c35d 100644
--- a/bookwyrm/utils/regex.py
+++ b/bookwyrm/utils/regex.py
@@ -1,10 +1,10 @@
-''' defining regexes for regularly used concepts '''
+""" defining regexes for regularly used concepts """
-domain = r'[\w_\-\.]+\.[a-z]{2,}'
-localname = r'@?[a-zA-Z_\-\.0-9]+'
-strict_localname = r'@[a-zA-Z_\-\.0-9]+'
-username = r'%s(@%s)?' % (localname, domain)
-strict_username = r'\B%s(@%s)?\b' % (strict_localname, domain)
-full_username = r'%s@%s\b' % (localname, domain)
+domain = r"[\w_\-\.]+\.[a-z]{2,}"
+localname = r"@?[a-zA-Z_\-\.0-9]+"
+strict_localname = r"@[a-zA-Z_\-\.0-9]+"
+username = r"%s(@%s)?" % (localname, domain)
+strict_username = r"\B%s(@%s)?\b" % (strict_localname, domain)
+full_username = r"%s@%s\b" % (localname, domain)
# should match (BookWyrm/1.0.0; or (BookWyrm/99.1.2;
-bookwyrm_user_agent = r'\(BookWyrm/[0-9]+\.[0-9]+\.[0-9]+;'
+bookwyrm_user_agent = r"\(BookWyrm/[0-9]+\.[0-9]+\.[0-9]+;"
diff --git a/bookwyrm/views/__init__.py b/bookwyrm/views/__init__.py
index 2c7cdc46..48da8ec1 100644
--- a/bookwyrm/views/__init__.py
+++ b/bookwyrm/views/__init__.py
@@ -1,4 +1,4 @@
-''' make sure all our nice views are available '''
+""" make sure all our nice views are available """
from .authentication import Login, Register, Logout
from .author import Author, EditAuthor
from .block import Block, unblock
@@ -31,3 +31,4 @@ from .site import Site
from .status import CreateStatus, DeleteStatus
from .updates import Updates
from .user import User, EditUser, Followers, Following
+from .isbn import Isbn
diff --git a/bookwyrm/views/authentication.py b/bookwyrm/views/authentication.py
index a74febca..a9147556 100644
--- a/bookwyrm/views/authentication.py
+++ b/bookwyrm/views/authentication.py
@@ -1,4 +1,4 @@
-''' class views for login/register views '''
+""" class views for login/register views """
from django.contrib.auth import authenticate, login, logout
from django.contrib.auth.decorators import login_required
from django.core.exceptions import PermissionDenied
@@ -6,6 +6,7 @@ from django.shortcuts import get_object_or_404, redirect
from django.template.response import TemplateResponse
from django.utils import timezone
from django.utils.decorators import method_decorator
+from django.views.decorators.csrf import csrf_exempt
from django.views import View
from bookwyrm import forms, models
@@ -13,58 +14,59 @@ from bookwyrm.settings import DOMAIN
# pylint: disable= no-self-use
+@method_decorator(csrf_exempt, name="dispatch")
class Login(View):
- ''' authenticate an existing user '''
+ """ authenticate an existing user """
+
def get(self, request):
- ''' login page '''
+ """ login page """
if request.user.is_authenticated:
- return redirect('/')
+ return redirect("/")
# sene user to the login page
data = {
- 'title': 'Login',
- 'login_form': forms.LoginForm(),
- 'register_form': forms.RegisterForm(),
+ "login_form": forms.LoginForm(),
+ "register_form": forms.RegisterForm(),
}
- return TemplateResponse(request, 'login.html', data)
+ return TemplateResponse(request, "login.html", data)
def post(self, request):
- ''' authentication action '''
+ """ authentication action """
+ if request.user.is_authenticated:
+ return redirect("/")
login_form = forms.LoginForm(request.POST)
- localname = login_form.data['localname']
- if '@' in localname: # looks like an email address to me
+ localname = login_form.data["localname"]
+ if "@" in localname: # looks like an email address to me
email = localname
try:
username = models.User.objects.get(email=email)
- except models.User.DoesNotExist: # maybe it's a full username?
+ except models.User.DoesNotExist: # maybe it's a full username?
username = localname
else:
- username = '%s@%s' % (localname, DOMAIN)
- password = login_form.data['password']
+ username = "%s@%s" % (localname, DOMAIN)
+ password = login_form.data["password"]
user = authenticate(request, username=username, password=password)
if user is not None:
# successful login
login(request, user)
user.last_active_date = timezone.now()
user.save(broadcast=False)
- return redirect(request.GET.get('next', '/'))
+ return redirect(request.GET.get("next", "/"))
# login errors
- login_form.non_field_errors = 'Username or password are incorrect'
+ login_form.non_field_errors = "Username or password are incorrect"
register_form = forms.RegisterForm()
- data = {
- 'login_form': login_form,
- 'register_form': register_form
- }
- return TemplateResponse(request, 'login.html', data)
+ data = {"login_form": login_form, "register_form": register_form}
+ return TemplateResponse(request, "login.html", data)
class Register(View):
- ''' register a user '''
+ """ register a user """
+
def post(self, request):
- ''' join the server '''
+ """ join the server """
if not models.SiteSettings.get().allow_registration:
- invite_code = request.POST.get('invite_code')
+ invite_code = request.POST.get("invite_code")
if not invite_code:
raise PermissionDenied
@@ -80,42 +82,43 @@ class Register(View):
if not form.is_valid():
errors = True
- localname = form.data['localname'].strip()
- email = form.data['email']
- password = form.data['password']
+ localname = form.data["localname"].strip()
+ email = form.data["email"]
+ password = form.data["password"]
# check localname and email uniqueness
if models.User.objects.filter(localname=localname).first():
- form.errors['localname'] = [
- 'User with this username already exists']
+ form.errors["localname"] = ["User with this username already exists"]
errors = True
if errors:
data = {
- 'login_form': forms.LoginForm(),
- 'register_form': form,
- 'invite': invite,
- 'valid': invite.valid() if invite else True,
+ "login_form": forms.LoginForm(),
+ "register_form": form,
+ "invite": invite,
+ "valid": invite.valid() if invite else True,
}
if invite:
- return TemplateResponse(request, 'invite.html', data)
- return TemplateResponse(request, 'login.html', data)
+ return TemplateResponse(request, "invite.html", data)
+ return TemplateResponse(request, "login.html", data)
- username = '%s@%s' % (localname, DOMAIN)
+ username = "%s@%s" % (localname, DOMAIN)
user = models.User.objects.create_user(
- username, email, password, localname=localname, local=True)
+ username, email, password, localname=localname, local=True
+ )
if invite:
invite.times_used += 1
invite.save()
login(request, user)
- return redirect('/')
+ return redirect("/")
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Logout(View):
- ''' log out '''
+ """ log out """
+
def get(self, request):
- ''' done with this place! outa here! '''
+ """ done with this place! outa here! """
logout(request)
- return redirect('/')
+ return redirect("/")
diff --git a/bookwyrm/views/author.py b/bookwyrm/views/author.py
index a1a37cc5..50a3588d 100644
--- a/bookwyrm/views/author.py
+++ b/bookwyrm/views/author.py
@@ -1,4 +1,4 @@
-''' the good people stuff! the authors! '''
+""" the good people stuff! the authors! """
from django.contrib.auth.decorators import login_required, permission_required
from django.db.models import Q
from django.shortcuts import get_object_or_404, redirect
@@ -13,52 +13,46 @@ from .helpers import is_api_request
# pylint: disable= no-self-use
class Author(View):
- ''' this person wrote a book '''
+ """ this person wrote a book """
+
def get(self, request, author_id):
- ''' landing page for an author '''
+ """ landing page for an author """
author = get_object_or_404(models.Author, id=author_id)
if is_api_request(request):
return ActivitypubResponse(author.to_activity())
books = models.Work.objects.filter(
- Q(authors=author) | Q(editions__authors=author)).distinct()
+ Q(authors=author) | Q(editions__authors=author)
+ ).distinct()
data = {
- 'title': author.name,
- 'author': author,
- 'books': [b.get_default_edition() for b in books],
+ "author": author,
+ "books": [b.get_default_edition() for b in books],
}
- return TemplateResponse(request, 'author.html', data)
+ return TemplateResponse(request, "author.html", data)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
@method_decorator(
- permission_required('bookwyrm.edit_book', raise_exception=True),
- name='dispatch')
+ permission_required("bookwyrm.edit_book", raise_exception=True), name="dispatch"
+)
class EditAuthor(View):
- ''' edit author info '''
+ """ edit author info """
+
def get(self, request, author_id):
- ''' info about a book '''
+ """ info about a book """
author = get_object_or_404(models.Author, id=author_id)
- data = {
- 'title': 'Edit Author',
- 'author': author,
- 'form': forms.AuthorForm(instance=author)
- }
- return TemplateResponse(request, 'edit_author.html', data)
+ data = {"author": author, "form": forms.AuthorForm(instance=author)}
+ return TemplateResponse(request, "edit_author.html", data)
def post(self, request, author_id):
- ''' edit a author cool '''
+ """ edit a author cool """
author = get_object_or_404(models.Author, id=author_id)
form = forms.AuthorForm(request.POST, request.FILES, instance=author)
if not form.is_valid():
- data = {
- 'title': 'Edit Author',
- 'author': author,
- 'form': form
- }
- return TemplateResponse(request, 'edit_author.html', data)
+ data = {"author": author, "form": form}
+ return TemplateResponse(request, "edit_author.html", data)
author = form.save()
- return redirect('/author/%s' % author.id)
+ return redirect("/author/%s" % author.id)
diff --git a/bookwyrm/views/block.py b/bookwyrm/views/block.py
index cb14aae3..6d6a8a58 100644
--- a/bookwyrm/views/block.py
+++ b/bookwyrm/views/block.py
@@ -1,4 +1,4 @@
-''' views for actions you can take in the application '''
+""" views for actions you can take in the application """
from django.contrib.auth.decorators import login_required
from django.http import HttpResponseNotFound
from django.shortcuts import get_object_or_404, redirect
@@ -10,26 +10,27 @@ from django.views.decorators.http import require_POST
from bookwyrm import models
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Block(View):
- ''' blocking users '''
+ """ blocking users """
+
def get(self, request):
- ''' list of blocked users? '''
- return TemplateResponse(
- request, 'preferences/blocks.html', {'title': 'Blocked Users'})
+ """ list of blocked users? """
+ return TemplateResponse(request, "preferences/blocks.html")
def post(self, request, user_id):
- ''' block a user '''
+ """ block a user """
to_block = get_object_or_404(models.User, id=user_id)
models.UserBlocks.objects.create(
- user_subject=request.user, user_object=to_block)
- return redirect('/preferences/block')
+ user_subject=request.user, user_object=to_block
+ )
+ return redirect("/preferences/block")
@require_POST
@login_required
def unblock(request, user_id):
- ''' undo a block '''
+ """ undo a block """
to_unblock = get_object_or_404(models.User, id=user_id)
try:
block = models.UserBlocks.objects.get(
@@ -39,4 +40,4 @@ def unblock(request, user_id):
except models.UserBlocks.DoesNotExist:
return HttpResponseNotFound()
block.delete()
- return redirect('/preferences/block')
+ return redirect("/preferences/block")
diff --git a/bookwyrm/views/books.py b/bookwyrm/views/books.py
index e05c4726..46df0483 100644
--- a/bookwyrm/views/books.py
+++ b/bookwyrm/views/books.py
@@ -1,4 +1,4 @@
-''' the good stuff! the books! '''
+""" the good stuff! the books! """
from django.core.paginator import Paginator
from django.contrib.auth.decorators import login_required, permission_required
from django.db import transaction
@@ -20,11 +20,12 @@ from .helpers import privacy_filter
# pylint: disable= no-self-use
class Book(View):
- ''' a book! this is the stuff '''
+ """ a book! this is the stuff """
+
def get(self, request, book_id):
- ''' info about a book '''
+ """ info about a book """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
@@ -50,30 +51,28 @@ class Book(View):
reviews = get_activity_feed(request.user, queryset=reviews)
# the reviews to show
- paginated = Paginator(reviews.exclude(
- Q(content__isnull=True) | Q(content='')
- ), PAGE_LENGTH)
+ paginated = Paginator(
+ reviews.exclude(Q(content__isnull=True) | Q(content="")), PAGE_LENGTH
+ )
reviews_page = paginated.page(page)
user_tags = readthroughs = user_shelves = other_edition_shelves = []
if request.user.is_authenticated:
user_tags = models.UserTag.objects.filter(
book=book, user=request.user
- ).values_list('tag__identifier', flat=True)
+ ).values_list("tag__identifier", flat=True)
readthroughs = models.ReadThrough.objects.filter(
user=request.user,
book=book,
- ).order_by('start_date')
+ ).order_by("start_date")
for readthrough in readthroughs:
- readthrough.progress_updates = \
- readthrough.progressupdate_set.all() \
- .order_by('-updated_date')
+ readthrough.progress_updates = (
+ readthrough.progressupdate_set.all().order_by("-updated_date")
+ )
- user_shelves = models.ShelfBook.objects.filter(
- user=request.user, book=book
- )
+ user_shelves = models.ShelfBook.objects.filter(user=request.user, book=book)
other_edition_shelves = models.ShelfBook.objects.filter(
~Q(book=book),
@@ -82,131 +81,124 @@ class Book(View):
)
data = {
- 'title': book.title,
- 'book': book,
- 'reviews': reviews_page,
- 'review_count': reviews.count(),
- 'ratings': reviews.filter(Q(content__isnull=True) | Q(content='')),
- 'rating': reviews.aggregate(Avg('rating'))['rating__avg'],
- 'tags': models.UserTag.objects.filter(book=book),
- 'lists': privacy_filter(
- request.user, book.list_set.all()
+ "book": book,
+ "reviews": reviews_page,
+ "review_count": reviews.count(),
+ "ratings": reviews.filter(Q(content__isnull=True) | Q(content="")),
+ "rating": reviews.aggregate(Avg("rating"))["rating__avg"],
+ "tags": models.UserTag.objects.filter(book=book),
+ "lists": privacy_filter(
+ request.user, book.list_set.filter(listitem__approved=True)
),
- 'user_tags': user_tags,
- 'user_shelves': user_shelves,
- 'other_edition_shelves': other_edition_shelves,
- 'readthroughs': readthroughs,
- 'path': '/book/%s' % book_id,
+ "user_tags": user_tags,
+ "user_shelves": user_shelves,
+ "other_edition_shelves": other_edition_shelves,
+ "readthroughs": readthroughs,
+ "path": "/book/%s" % book_id,
}
- return TemplateResponse(request, 'book.html', data)
+ return TemplateResponse(request, "book.html", data)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
@method_decorator(
- permission_required('bookwyrm.edit_book', raise_exception=True),
- name='dispatch')
+ permission_required("bookwyrm.edit_book", raise_exception=True), name="dispatch"
+)
class EditBook(View):
- ''' edit a book '''
+ """ edit a book """
+
def get(self, request, book_id):
- ''' info about a book '''
+ """ info about a book """
book = get_edition(book_id)
if not book.description:
book.description = book.parent_work.description
- data = {
- 'title': 'Edit Book',
- 'book': book,
- 'form': forms.EditionForm(instance=book)
- }
- return TemplateResponse(request, 'edit_book.html', data)
+ data = {"book": book, "form": forms.EditionForm(instance=book)}
+ return TemplateResponse(request, "edit_book.html", data)
def post(self, request, book_id):
- ''' edit a book cool '''
+ """ edit a book cool """
book = get_object_or_404(models.Edition, id=book_id)
form = forms.EditionForm(request.POST, request.FILES, instance=book)
if not form.is_valid():
- data = {
- 'title': 'Edit Book',
- 'book': book,
- 'form': form
- }
- return TemplateResponse(request, 'edit_book.html', data)
+ data = {"book": book, "form": form}
+ return TemplateResponse(request, "edit_book.html", data)
book = form.save()
- return redirect('/book/%s' % book.id)
+ return redirect("/book/%s" % book.id)
class Editions(View):
- ''' list of editions '''
+ """ list of editions """
+
def get(self, request, book_id):
- ''' list of editions of a book '''
+ """ list of editions of a book """
work = get_object_or_404(models.Work, id=book_id)
if is_api_request(request):
return ActivitypubResponse(work.to_edition_list(**request.GET))
data = {
- 'title': 'Editions of %s' % work.title,
- 'editions': work.editions.order_by('-edition_rank').all(),
- 'work': work,
+ "editions": work.editions.order_by("-edition_rank").all(),
+ "work": work,
}
- return TemplateResponse(request, 'editions.html', data)
+ return TemplateResponse(request, "editions.html", data)
@login_required
@require_POST
def upload_cover(request, book_id):
- ''' upload a new cover '''
+ """ upload a new cover """
book = get_object_or_404(models.Edition, id=book_id)
form = forms.CoverForm(request.POST, request.FILES, instance=book)
if not form.is_valid():
- return redirect('/book/%d' % book.id)
+ return redirect("/book/%d" % book.id)
- book.cover = form.files['cover']
+ book.last_edited_by = request.user
+ book.cover = form.files["cover"]
book.save()
- return redirect('/book/%s' % book.id)
+ return redirect("/book/%s" % book.id)
@login_required
@require_POST
-@permission_required('bookwyrm.edit_book', raise_exception=True)
+@permission_required("bookwyrm.edit_book", raise_exception=True)
def add_description(request, book_id):
- ''' upload a new cover '''
- if not request.method == 'POST':
- return redirect('/')
+ """ upload a new cover """
+ if not request.method == "POST":
+ return redirect("/")
book = get_object_or_404(models.Edition, id=book_id)
- description = request.POST.get('description')
+ description = request.POST.get("description")
book.description = description
+ book.last_edited_by = request.user
book.save()
- return redirect('/book/%s' % book.id)
+ return redirect("/book/%s" % book.id)
@require_POST
def resolve_book(request):
- ''' figure out the local path to a book from a remote_id '''
- remote_id = request.POST.get('remote_id')
+ """ figure out the local path to a book from a remote_id """
+ remote_id = request.POST.get("remote_id")
connector = connector_manager.get_or_create_connector(remote_id)
book = connector.get_or_create_book(remote_id)
- return redirect('/book/%d' % book.id)
+ return redirect("/book/%d" % book.id)
@login_required
@require_POST
@transaction.atomic
def switch_edition(request):
- ''' switch your copy of a book to a different edition '''
- edition_id = request.POST.get('edition')
+ """ switch your copy of a book to a different edition """
+ edition_id = request.POST.get("edition")
new_edition = get_object_or_404(models.Edition, id=edition_id)
shelfbooks = models.ShelfBook.objects.filter(
- book__parent_work=new_edition.parent_work,
- shelf__user=request.user
+ book__parent_work=new_edition.parent_work, shelf__user=request.user
)
for shelfbook in shelfbooks.all():
with transaction.atomic():
@@ -214,16 +206,15 @@ def switch_edition(request):
created_date=shelfbook.created_date,
user=shelfbook.user,
shelf=shelfbook.shelf,
- book=new_edition
+ book=new_edition,
)
shelfbook.delete()
readthroughs = models.ReadThrough.objects.filter(
- book__parent_work=new_edition.parent_work,
- user=request.user
+ book__parent_work=new_edition.parent_work, user=request.user
)
for readthrough in readthroughs.all():
readthrough.book = new_edition
readthrough.save()
- return redirect('/book/%d' % new_edition.id)
+ return redirect("/book/%d" % new_edition.id)
diff --git a/bookwyrm/views/error.py b/bookwyrm/views/error.py
index 9eabe29f..9bd94ca3 100644
--- a/bookwyrm/views/error.py
+++ b/bookwyrm/views/error.py
@@ -1,13 +1,12 @@
-''' something has gone amiss '''
+""" something has gone amiss """
from django.template.response import TemplateResponse
+
def server_error_page(request):
- ''' 500 errors '''
- return TemplateResponse(
- request, 'error.html', {'title': 'Oops!'}, status=500)
+ """ 500 errors """
+ return TemplateResponse(request, "error.html", status=500)
def not_found_page(request, _):
- ''' 404s '''
- return TemplateResponse(
- request, 'notfound.html', {'title': 'Not found'}, status=404)
+ """ 404s """
+ return TemplateResponse(request, "notfound.html", status=404)
diff --git a/bookwyrm/views/federation.py b/bookwyrm/views/federation.py
index 0bd14dab..e9c0466e 100644
--- a/bookwyrm/views/federation.py
+++ b/bookwyrm/views/federation.py
@@ -1,4 +1,4 @@
-''' manage federated servers '''
+""" manage federated servers """
from django.contrib.auth.decorators import login_required, permission_required
from django.template.response import TemplateResponse
from django.utils.decorators import method_decorator
@@ -8,17 +8,16 @@ from bookwyrm import models
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
@method_decorator(
- permission_required('bookwyrm.control_federation', raise_exception=True),
- name='dispatch')
+ permission_required("bookwyrm.control_federation", raise_exception=True),
+ name="dispatch",
+)
class Federation(View):
- ''' what servers do we federate with '''
+ """ what servers do we federate with """
+
def get(self, request):
- ''' edit form '''
+ """ edit form """
servers = models.FederatedServer.objects.all()
- data = {
- 'title': 'Federated Servers',
- 'servers': servers
- }
- return TemplateResponse(request, 'settings/federation.html', data)
+ data = {"servers": servers}
+ return TemplateResponse(request, "settings/federation.html", data)
diff --git a/bookwyrm/views/feed.py b/bookwyrm/views/feed.py
index 3a2805b4..d08c9a42 100644
--- a/bookwyrm/views/feed.py
+++ b/bookwyrm/views/feed.py
@@ -1,4 +1,4 @@
-''' non-interactive pages '''
+""" non-interactive pages """
from django.contrib.auth.decorators import login_required
from django.core.paginator import Paginator
from django.db.models import Q
@@ -6,6 +6,7 @@ from django.http import HttpResponseNotFound
from django.template.response import TemplateResponse
from django.utils import timezone
from django.utils.decorators import method_decorator
+from django.utils.translation import gettext as _
from django.views import View
from bookwyrm import forms, models
@@ -16,45 +17,54 @@ from .helpers import is_api_request, is_bookwyrm_request, object_visible_to_user
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Feed(View):
- ''' activity stream '''
+ """ activity stream """
+
def get(self, request, tab):
- ''' user's homepage with activity feed '''
+ """ user's homepage with activity feed """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
- if tab == 'home':
+ if tab == "home":
+ activities = get_activity_feed(request.user, following_only=True)
+ tab_title = _("Home")
+ elif tab == "local":
activities = get_activity_feed(
- request.user, following_only=True)
- elif tab == 'local':
- activities = get_activity_feed(
- request.user, privacy=['public', 'followers'], local_only=True)
+ request.user, privacy=["public", "followers"], local_only=True
+ )
+ tab_title = _("Local")
else:
activities = get_activity_feed(
- request.user, privacy=['public', 'followers'])
+ request.user, privacy=["public", "followers"]
+ )
+ tab_title = _("Federated")
paginated = Paginator(activities, PAGE_LENGTH)
- data = {**feed_page_data(request.user), **{
- 'title': 'Updates Feed',
- 'user': request.user,
- 'activities': paginated.page(page),
- 'tab': tab,
- 'goal_form': forms.GoalForm(),
- 'path': '/%s' % tab,
- }}
- return TemplateResponse(request, 'feed/feed.html', data)
+ data = {
+ **feed_page_data(request.user),
+ **{
+ "user": request.user,
+ "activities": paginated.page(page),
+ "tab": tab,
+ "tab_title": tab_title,
+ "goal_form": forms.GoalForm(),
+ "path": "/%s" % tab,
+ },
+ }
+ return TemplateResponse(request, "feed/feed.html", data)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class DirectMessage(View):
- ''' dm view '''
+ """ dm view """
+
def get(self, request, username=None):
- ''' like a feed but for dms only '''
+ """ like a feed but for dms only """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
@@ -70,28 +80,33 @@ class DirectMessage(View):
queryset = queryset.filter(Q(user=user) | Q(mention_users=user))
activities = get_activity_feed(
- request.user, privacy=['direct'], queryset=queryset)
+ request.user, privacy=["direct"], queryset=queryset
+ )
paginated = Paginator(activities, PAGE_LENGTH)
activity_page = paginated.page(page)
- data = {**feed_page_data(request.user), **{
- 'title': 'Direct Messages',
- 'user': request.user,
- 'partner': user,
- 'activities': activity_page,
- 'path': '/direct-messages',
- }}
- return TemplateResponse(request, 'feed/direct_messages.html', data)
+ data = {
+ **feed_page_data(request.user),
+ **{
+ "user": request.user,
+ "partner": user,
+ "activities": activity_page,
+ "path": "/direct-messages",
+ },
+ }
+ return TemplateResponse(request, "feed/direct_messages.html", data)
class Status(View):
- ''' get posting '''
+ """ get posting """
+
def get(self, request, username, status_id):
- ''' display a particular status (and replies, etc) '''
+ """ display a particular status (and replies, etc) """
try:
user = get_user_from_username(request.user, username)
status = models.Status.objects.select_subclasses().get(
- id=status_id, deleted=False)
+ id=status_id, deleted=False
+ )
except ValueError:
return HttpResponseNotFound()
@@ -105,19 +120,23 @@ class Status(View):
if is_api_request(request):
return ActivitypubResponse(
- status.to_activity(pure=not is_bookwyrm_request(request)))
+ status.to_activity(pure=not is_bookwyrm_request(request))
+ )
- data = {**feed_page_data(request.user), **{
- 'title': 'Status by %s' % user.username,
- 'status': status,
- }}
- return TemplateResponse(request, 'feed/status.html', data)
+ data = {
+ **feed_page_data(request.user),
+ **{
+ "status": status,
+ },
+ }
+ return TemplateResponse(request, "feed/status.html", data)
class Replies(View):
- ''' replies page (a json view of status) '''
+ """ replies page (a json view of status) """
+
def get(self, request, username, status_id):
- ''' ordered collection of replies to a status '''
+ """ ordered collection of replies to a status """
# the html view is the same as Status
if not is_api_request(request):
status_view = Status.as_view()
@@ -132,40 +151,39 @@ class Replies(View):
def feed_page_data(user):
- ''' info we need for every feed page '''
+ """ info we need for every feed page """
if not user.is_authenticated:
return {}
- goal = models.AnnualGoal.objects.filter(
- user=user, year=timezone.now().year
- ).first()
+ goal = models.AnnualGoal.objects.filter(user=user, year=timezone.now().year).first()
return {
- 'suggested_books': get_suggested_books(user),
- 'goal': goal,
- 'goal_form': forms.GoalForm(),
+ "suggested_books": get_suggested_books(user),
+ "goal": goal,
+ "goal_form": forms.GoalForm(),
}
+
def get_suggested_books(user, max_books=5):
- ''' helper to get a user's recent books '''
+ """ helper to get a user's recent books """
book_count = 0
- preset_shelves = [
- ('reading', max_books), ('read', 2), ('to-read', max_books)
- ]
+ preset_shelves = [("reading", max_books), ("read", 2), ("to-read", max_books)]
suggested_books = []
for (preset, shelf_max) in preset_shelves:
- limit = shelf_max if shelf_max < (max_books - book_count) \
- else max_books - book_count
+ limit = (
+ shelf_max
+ if shelf_max < (max_books - book_count)
+ else max_books - book_count
+ )
shelf = user.shelf_set.get(identifier=preset)
- shelf_books = shelf.shelfbook_set.order_by(
- '-updated_date'
- ).all()[:limit]
+ shelf_books = shelf.shelfbook_set.order_by("-updated_date").all()[:limit]
if not shelf_books:
continue
shelf_preview = {
- 'name': shelf.name,
- 'books': [s.book for s in shelf_books]
+ "name": shelf.name,
+ "identifier": shelf.identifier,
+ "books": [s.book for s in shelf_books],
}
suggested_books.append(shelf_preview)
- book_count += len(shelf_preview['books'])
+ book_count += len(shelf_preview["books"])
return suggested_books
diff --git a/bookwyrm/views/follow.py b/bookwyrm/views/follow.py
index 4c69890c..515bf325 100644
--- a/bookwyrm/views/follow.py
+++ b/bookwyrm/views/follow.py
@@ -1,4 +1,4 @@
-''' views for actions you can take in the application '''
+""" views for actions you can take in the application """
from django.contrib.auth.decorators import login_required
from django.db import IntegrityError
from django.http import HttpResponseBadRequest
@@ -8,11 +8,12 @@ from django.views.decorators.http import require_POST
from bookwyrm import models
from .helpers import get_user_from_username
+
@login_required
@require_POST
def follow(request):
- ''' follow another user, here or abroad '''
- username = request.POST['user']
+ """ follow another user, here or abroad """
+ username = request.POST["user"]
try:
to_follow = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -32,16 +33,15 @@ def follow(request):
@login_required
@require_POST
def unfollow(request):
- ''' unfollow a user '''
- username = request.POST['user']
+ """ unfollow a user """
+ username = request.POST["user"]
try:
to_unfollow = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
return HttpResponseBadRequest()
models.UserFollows.objects.get(
- user_subject=request.user,
- user_object=to_unfollow
+ user_subject=request.user, user_object=to_unfollow
).delete()
return redirect(to_unfollow.local_path)
@@ -49,8 +49,8 @@ def unfollow(request):
@login_required
@require_POST
def accept_follow_request(request):
- ''' a user accepts a follow request '''
- username = request.POST['user']
+ """ a user accepts a follow request """
+ username = request.POST["user"]
try:
requester = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -58,8 +58,7 @@ def accept_follow_request(request):
try:
follow_request = models.UserFollowRequest.objects.get(
- user_subject=requester,
- user_object=request.user
+ user_subject=requester, user_object=request.user
)
except models.UserFollowRequest.DoesNotExist:
# Request already dealt with.
@@ -72,8 +71,8 @@ def accept_follow_request(request):
@login_required
@require_POST
def delete_follow_request(request):
- ''' a user rejects a follow request '''
- username = request.POST['user']
+ """ a user rejects a follow request """
+ username = request.POST["user"]
try:
requester = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -81,11 +80,10 @@ def delete_follow_request(request):
try:
follow_request = models.UserFollowRequest.objects.get(
- user_subject=requester,
- user_object=request.user
+ user_subject=requester, user_object=request.user
)
except models.UserFollowRequest.DoesNotExist:
return HttpResponseBadRequest()
follow_request.delete()
- return redirect('/user/%s' % request.user.localname)
+ return redirect("/user/%s" % request.user.localname)
diff --git a/bookwyrm/views/goal.py b/bookwyrm/views/goal.py
index 97f13913..4e3a9534 100644
--- a/bookwyrm/views/goal.py
+++ b/bookwyrm/views/goal.py
@@ -1,4 +1,4 @@
-''' non-interactive pages '''
+""" non-interactive pages """
from django.contrib.auth.decorators import login_required
from django.http import HttpResponseNotFound
from django.shortcuts import redirect
@@ -13,16 +13,15 @@ from .helpers import get_user_from_username, object_visible_to_user
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Goal(View):
- ''' track books for the year '''
+ """ track books for the year """
+
def get(self, request, username, year):
- ''' reading goal page '''
+ """ reading goal page """
user = get_user_from_username(request.user, username)
year = int(year)
- goal = models.AnnualGoal.objects.filter(
- year=year, user=user
- ).first()
+ goal = models.AnnualGoal.objects.filter(year=year, user=user).first()
if not goal and user != request.user:
return HttpResponseNotFound()
@@ -30,44 +29,39 @@ class Goal(View):
return HttpResponseNotFound()
data = {
- 'title': '%s\'s %d Reading' % (user.display_name, year),
- 'goal_form': forms.GoalForm(instance=goal),
- 'goal': goal,
- 'user': user,
- 'year': year,
- 'is_self': request.user == user,
+ "goal_form": forms.GoalForm(instance=goal),
+ "goal": goal,
+ "user": user,
+ "year": year,
+ "is_self": request.user == user,
}
- return TemplateResponse(request, 'goal.html', data)
-
+ return TemplateResponse(request, "goal.html", data)
def post(self, request, username, year):
- ''' update or create an annual goal '''
+ """ update or create an annual goal """
user = get_user_from_username(request.user, username)
if user != request.user:
return HttpResponseNotFound()
year = int(year)
- goal = models.AnnualGoal.objects.filter(
- year=year, user=request.user
- ).first()
+ goal = models.AnnualGoal.objects.filter(year=year, user=request.user).first()
form = forms.GoalForm(request.POST, instance=goal)
if not form.is_valid():
data = {
- 'title': '%s\'s %d Reading' % (request.user.display_name, year),
- 'goal_form': form,
- 'goal': goal,
- 'year': year,
+ "goal_form": form,
+ "goal": goal,
+ "year": year,
}
- return TemplateResponse(request, 'goal.html', data)
+ return TemplateResponse(request, "goal.html", data)
goal = form.save()
- if request.POST.get('post-status'):
+ if request.POST.get("post-status"):
# create status, if appropraite
- template = get_template('snippets/generated_status/goal.html')
+ template = get_template("snippets/generated_status/goal.html")
create_generated_note(
request.user,
- template.render({'goal': goal, 'user': request.user}).strip(),
- privacy=goal.privacy
+ template.render({"goal": goal, "user": request.user}).strip(),
+ privacy=goal.privacy,
)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
diff --git a/bookwyrm/views/helpers.py b/bookwyrm/views/helpers.py
index 64f0fc26..20332cbd 100644
--- a/bookwyrm/views/helpers.py
+++ b/bookwyrm/views/helpers.py
@@ -1,4 +1,4 @@
-''' helper functions used in various views '''
+""" helper functions used in various views """
import re
from requests import HTTPError
from django.core.exceptions import FieldError
@@ -11,7 +11,7 @@ from bookwyrm.utils import regex
def get_user_from_username(viewer, username):
- ''' helper function to resolve a localname or a username to a user '''
+ """ helper function to resolve a localname or a username to a user """
# raises DoesNotExist if user is now found
try:
return models.User.viewer_aware_objects(viewer).get(localname=username)
@@ -20,22 +20,20 @@ def get_user_from_username(viewer, username):
def is_api_request(request):
- ''' check whether a request is asking for html or data '''
- return 'json' in request.headers.get('Accept') or \
- request.path[-5:] == '.json'
+ """ check whether a request is asking for html or data """
+ return "json" in request.headers.get("Accept") or request.path[-5:] == ".json"
def is_bookwyrm_request(request):
- ''' check if the request is coming from another bookwyrm instance '''
- user_agent = request.headers.get('User-Agent')
- if user_agent is None or \
- re.search(regex.bookwyrm_user_agent, user_agent) is None:
+ """ check if the request is coming from another bookwyrm instance """
+ user_agent = request.headers.get("User-Agent")
+ if user_agent is None or re.search(regex.bookwyrm_user_agent, user_agent) is None:
return False
return True
def object_visible_to_user(viewer, obj):
- ''' is a user authorized to view an object? '''
+ """ is a user authorized to view an object? """
if not obj:
return False
@@ -44,37 +42,32 @@ def object_visible_to_user(viewer, obj):
return False
# you can see your own posts and any public or unlisted posts
- if viewer == obj.user or obj.privacy in ['public', 'unlisted']:
+ if viewer == obj.user or obj.privacy in ["public", "unlisted"]:
return True
# you can see the followers only posts of people you follow
- if obj.privacy == 'followers' and \
- obj.user.followers.filter(id=viewer.id).first():
+ if obj.privacy == "followers" and obj.user.followers.filter(id=viewer.id).first():
return True
# you can see dms you are tagged in
if isinstance(obj, models.Status):
- if obj.privacy == 'direct' and \
- obj.mention_users.filter(id=viewer.id).first():
+ if obj.privacy == "direct" and obj.mention_users.filter(id=viewer.id).first():
return True
return False
def privacy_filter(viewer, queryset, privacy_levels=None, following_only=False):
- ''' filter objects that have "user" and "privacy" fields '''
- privacy_levels = privacy_levels or \
- ['public', 'unlisted', 'followers', 'direct']
+ """ filter objects that have "user" and "privacy" fields """
+ privacy_levels = privacy_levels or ["public", "unlisted", "followers", "direct"]
# exclude blocks from both directions
if not viewer.is_anonymous:
blocked = models.User.objects.filter(id__in=viewer.blocks.all()).all()
- queryset = queryset.exclude(
- Q(user__in=blocked) | Q(user__blocks=viewer))
+ queryset = queryset.exclude(Q(user__in=blocked) | Q(user__blocks=viewer))
# you can't see followers only or direct messages if you're not logged in
if viewer.is_anonymous:
- privacy_levels = [p for p in privacy_levels if \
- not p in ['followers', 'direct']]
+ privacy_levels = [p for p in privacy_levels if not p in ["followers", "direct"]]
# filter to only privided privacy levels
queryset = queryset.filter(privacy__in=privacy_levels)
@@ -82,53 +75,48 @@ def privacy_filter(viewer, queryset, privacy_levels=None, following_only=False):
# only include statuses the user follows
if following_only:
queryset = queryset.exclude(
- ~Q(# remove everythign except
- Q(user__in=viewer.following.all()) | # user following
- Q(user=viewer) |# is self
- Q(mention_users=viewer)# mentions user
+ ~Q( # remove everythign except
+ Q(user__in=viewer.following.all())
+ | Q(user=viewer) # user following
+ | Q(mention_users=viewer) # is self # mentions user
),
)
# exclude followers-only statuses the user doesn't follow
- elif 'followers' in privacy_levels:
+ elif "followers" in privacy_levels:
queryset = queryset.exclude(
- ~Q(# user isn't following and it isn't their own status
+ ~Q( # user isn't following and it isn't their own status
Q(user__in=viewer.following.all()) | Q(user=viewer)
),
- privacy='followers' # and the status is followers only
+ privacy="followers", # and the status is followers only
)
# exclude direct messages not intended for the user
- if 'direct' in privacy_levels:
+ if "direct" in privacy_levels:
try:
queryset = queryset.exclude(
- ~Q(
- Q(user=viewer) | Q(mention_users=viewer)
- ), privacy='direct'
+ ~Q(Q(user=viewer) | Q(mention_users=viewer)), privacy="direct"
)
except FieldError:
- queryset = queryset.exclude(
- ~Q(user=viewer), privacy='direct'
- )
+ queryset = queryset.exclude(~Q(user=viewer), privacy="direct")
return queryset
def get_activity_feed(
- user, privacy=None, local_only=False, following_only=False,
- queryset=None):
- ''' get a filtered queryset of statuses '''
+ user, privacy=None, local_only=False, following_only=False, queryset=None
+):
+ """ get a filtered queryset of statuses """
if queryset is None:
queryset = models.Status.objects.select_subclasses()
# exclude deleted
- queryset = queryset.exclude(deleted=True).order_by('-published_date')
+ queryset = queryset.exclude(deleted=True).order_by("-published_date")
# apply privacy filters
- queryset = privacy_filter(
- user, queryset, privacy, following_only=following_only)
+ queryset = privacy_filter(user, queryset, privacy, following_only=following_only)
# only show dms if we only want dms
- if privacy == ['direct']:
+ if privacy == ["direct"]:
# dms are direct statuses not related to books
queryset = queryset.filter(
review__isnull=True,
@@ -143,7 +131,7 @@ def get_activity_feed(
comment__isnull=True,
quotation__isnull=True,
generatednote__isnull=True,
- privacy='direct'
+ privacy="direct",
)
except FieldError:
# if we're looking at a subtype of Status (like Review)
@@ -163,36 +151,35 @@ def get_activity_feed(
def handle_remote_webfinger(query):
- ''' webfingerin' other servers '''
+ """ webfingerin' other servers """
user = None
# usernames could be @user@domain or user@domain
if not query:
return None
- if query[0] == '@':
+ if query[0] == "@":
query = query[1:]
try:
- domain = query.split('@')[1]
+ domain = query.split("@")[1]
except IndexError:
return None
try:
user = models.User.objects.get(username=query)
except models.User.DoesNotExist:
- url = 'https://%s/.well-known/webfinger?resource=acct:%s' % \
- (domain, query)
+ url = "https://%s/.well-known/webfinger?resource=acct:%s" % (domain, query)
try:
data = get_data(url)
except (ConnectorException, HTTPError):
return None
- for link in data.get('links'):
- if link.get('rel') == 'self':
+ for link in data.get("links"):
+ if link.get("rel") == "self":
try:
user = activitypub.resolve_remote_id(
- link['href'], model=models.User
+ link["href"], model=models.User
)
except KeyError:
return None
@@ -200,7 +187,7 @@ def handle_remote_webfinger(query):
def get_edition(book_id):
- ''' look up a book in the db and return an edition '''
+ """ look up a book in the db and return an edition """
book = models.Book.objects.select_subclasses().get(id=book_id)
if isinstance(book, models.Work):
book = book.get_default_edition()
@@ -208,29 +195,24 @@ def get_edition(book_id):
def handle_reading_status(user, shelf, book, privacy):
- ''' post about a user reading a book '''
+ """ post about a user reading a book """
# tell the world about this cool thing that happened
try:
message = {
- 'to-read': 'wants to read',
- 'reading': 'started reading',
- 'read': 'finished reading'
+ "to-read": "wants to read",
+ "reading": "started reading",
+ "read": "finished reading",
}[shelf.identifier]
except KeyError:
# it's a non-standard shelf, don't worry about it
return
- status = create_generated_note(
- user,
- message,
- mention_books=[book],
- privacy=privacy
- )
+ status = create_generated_note(user, message, mention_books=[book], privacy=privacy)
status.save()
def is_blocked(viewer, user):
- ''' is this viewer blocked by the user? '''
+ """ is this viewer blocked by the user? """
if viewer.is_authenticated and viewer in user.blocks.all():
return True
return False
diff --git a/bookwyrm/views/import_data.py b/bookwyrm/views/import_data.py
index 675cae3d..8f9ea27f 100644
--- a/bookwyrm/views/import_data.py
+++ b/bookwyrm/views/import_data.py
@@ -1,4 +1,4 @@
-''' import books from another app '''
+""" import books from another app """
from io import TextIOWrapper
from django.contrib.auth.decorators import login_required
@@ -13,28 +13,33 @@ from bookwyrm import forms, goodreads_import, librarything_import, models
from bookwyrm.tasks import app
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Import(View):
- ''' import view '''
+ """ import view """
+
def get(self, request):
- ''' load import page '''
- return TemplateResponse(request, 'import.html', {
- 'title': 'Import Books',
- 'import_form': forms.ImportForm(),
- 'jobs': models.ImportJob.
- objects.filter(user=request.user).order_by('-created_date'),
- })
+ """ load import page """
+ return TemplateResponse(
+ request,
+ "import.html",
+ {
+ "import_form": forms.ImportForm(),
+ "jobs": models.ImportJob.objects.filter(user=request.user).order_by(
+ "-created_date"
+ ),
+ },
+ )
def post(self, request):
- ''' ingest a goodreads csv '''
+ """ ingest a goodreads csv """
form = forms.ImportForm(request.POST, request.FILES)
if form.is_valid():
- include_reviews = request.POST.get('include_reviews') == 'on'
- privacy = request.POST.get('privacy')
- source = request.POST.get('source')
+ include_reviews = request.POST.get("include_reviews") == "on"
+ privacy = request.POST.get("privacy")
+ source = request.POST.get("source")
importer = None
- if source == 'LibraryThing':
+ if source == "LibraryThing":
importer = librarything_import.LibrarythingImporter()
else:
# Default : GoodReads
@@ -44,45 +49,44 @@ class Import(View):
job = importer.create_job(
request.user,
TextIOWrapper(
- request.FILES['csv_file'],
- encoding=importer.encoding),
+ request.FILES["csv_file"], encoding=importer.encoding
+ ),
include_reviews,
privacy,
)
except (UnicodeDecodeError, ValueError):
- return HttpResponseBadRequest('Not a valid csv file')
+ return HttpResponseBadRequest("Not a valid csv file")
importer.start_import(job)
- return redirect('/import/%d' % job.id)
+ return redirect("/import/%d" % job.id)
return HttpResponseBadRequest()
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class ImportStatus(View):
- ''' status of an existing import '''
+ """ status of an existing import """
+
def get(self, request, job_id):
- ''' status of an import job '''
+ """ status of an import job """
job = models.ImportJob.objects.get(id=job_id)
if job.user != request.user:
raise PermissionDenied
task = app.AsyncResult(job.task_id)
- items = job.items.order_by('index').all()
+ items = job.items.order_by("index").all()
failed_items = [i for i in items if i.fail_reason]
items = [i for i in items if not i.fail_reason]
- return TemplateResponse(request, 'import_status.html', {
- 'title': 'Import Status',
- 'job': job,
- 'items': items,
- 'failed_items': failed_items,
- 'task': task
- })
+ return TemplateResponse(
+ request,
+ "import_status.html",
+ {"job": job, "items": items, "failed_items": failed_items, "task": task},
+ )
def post(self, request, job_id):
- ''' retry lines from an import '''
+ """ retry lines from an import """
job = get_object_or_404(models.ImportJob, id=job_id)
items = []
- for item in request.POST.getlist('import_item'):
+ for item in request.POST.getlist("import_item"):
items.append(get_object_or_404(models.ImportItem, id=item))
job = goodreads_import.create_retry_job(
@@ -91,4 +95,4 @@ class ImportStatus(View):
items,
)
goodreads_import.start_import(job)
- return redirect('/import/%d' % job.id)
+ return redirect("/import/%d" % job.id)
diff --git a/bookwyrm/views/inbox.py b/bookwyrm/views/inbox.py
index 4da4e5b6..34bd2e1c 100644
--- a/bookwyrm/views/inbox.py
+++ b/bookwyrm/views/inbox.py
@@ -1,4 +1,4 @@
-''' incoming activities '''
+""" incoming activities """
import json
from urllib.parse import urldefrag
@@ -14,13 +14,14 @@ from bookwyrm.tasks import app
from bookwyrm.signatures import Signature
-@method_decorator(csrf_exempt, name='dispatch')
+@method_decorator(csrf_exempt, name="dispatch")
# pylint: disable=no-self-use
class Inbox(View):
- ''' requests sent by outside servers'''
+ """ requests sent by outside servers"""
+
def post(self, request, username=None):
- ''' only works as POST request '''
- # first let's do some basic checks to see if this is legible
+ """ only works as POST request """
+ # make sure the user's inbox even exists
if username:
try:
models.User.objects.get(localname=username)
@@ -33,28 +34,29 @@ class Inbox(View):
except json.decoder.JSONDecodeError:
return HttpResponseBadRequest()
+ if (
+ not "object" in activity_json
+ or not "type" in activity_json
+ or not activity_json["type"] in activitypub.activity_objects
+ ):
+ return HttpResponseNotFound()
+
# verify the signature
if not has_valid_signature(request, activity_json):
- if activity_json['type'] == 'Delete':
+ if activity_json["type"] == "Delete":
# Pretend that unauth'd deletes succeed. Auth may be failing
# because the resource or owner of the resource might have
# been deleted.
return HttpResponse()
return HttpResponse(status=401)
- # just some quick smell tests before we try to parse the json
- if not 'object' in activity_json or \
- not 'type' in activity_json or \
- not activity_json['type'] in activitypub.activity_objects:
- return HttpResponseNotFound()
-
activity_task.delay(activity_json)
return HttpResponse()
@app.task
def activity_task(activity_json):
- ''' do something with this json we think is legit '''
+ """ do something with this json we think is legit """
# lets see if the activitypub module can make sense of this json
try:
activity = activitypub.parse(activity_json)
@@ -63,20 +65,23 @@ def activity_task(activity_json):
# cool that worked, now we should do the action described by the type
# (create, update, delete, etc)
- activity.action()
+ try:
+ activity.action()
+ except activitypub.ActivitySerializerError:
+ # this is raised if the activity is discarded
+ return
def has_valid_signature(request, activity):
- ''' verify incoming signature '''
+ """ verify incoming signature """
try:
signature = Signature.parse(request)
key_actor = urldefrag(signature.key_id).url
- if key_actor != activity.get('actor'):
+ if key_actor != activity.get("actor"):
raise ValueError("Wrong actor created signature.")
- remote_user = activitypub.resolve_remote_id(
- key_actor, model=models.User)
+ remote_user = activitypub.resolve_remote_id(key_actor, model=models.User)
if not remote_user:
return False
@@ -88,7 +93,7 @@ def has_valid_signature(request, activity):
remote_user.remote_id, model=models.User, refresh=True
)
if remote_user.key_pair.public_key == old_key:
- raise # Key unchanged.
+ raise # Key unchanged.
signature.verify(remote_user.key_pair.public_key, request)
except (ValueError, requests.exceptions.HTTPError):
return False
diff --git a/bookwyrm/views/interaction.py b/bookwyrm/views/interaction.py
index a7fcc231..e337f2ef 100644
--- a/bookwyrm/views/interaction.py
+++ b/bookwyrm/views/interaction.py
@@ -1,4 +1,4 @@
-''' boosts and favs '''
+""" boosts and favs """
from django.db import IntegrityError
from django.contrib.auth.decorators import login_required
from django.http import HttpResponseBadRequest, HttpResponseNotFound
@@ -10,75 +10,74 @@ from bookwyrm import models
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Favorite(View):
- ''' like a status '''
+ """ like a status """
+
def post(self, request, status_id):
- ''' create a like '''
+ """ create a like """
status = models.Status.objects.get(id=status_id)
try:
- models.Favorite.objects.create(
- status=status,
- user=request.user
- )
+ models.Favorite.objects.create(status=status, user=request.user)
except IntegrityError:
# you already fav'ed that
return HttpResponseBadRequest()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Unfavorite(View):
- ''' take back a fav '''
+ """ take back a fav """
+
def post(self, request, status_id):
- ''' unlike a status '''
+ """ unlike a status """
status = models.Status.objects.get(id=status_id)
try:
- favorite = models.Favorite.objects.get(
- status=status,
- user=request.user
- )
+ favorite = models.Favorite.objects.get(status=status, user=request.user)
except models.Favorite.DoesNotExist:
# can't find that status, idk
return HttpResponseNotFound()
favorite.delete()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Boost(View):
- ''' boost a status '''
+ """ boost a status """
+
def post(self, request, status_id):
- ''' boost a status '''
+ """ boost a status """
status = models.Status.objects.get(id=status_id)
# is it boostable?
if not status.boostable:
return HttpResponseBadRequest()
if models.Boost.objects.filter(
- boosted_status=status, user=request.user).exists():
+ boosted_status=status, user=request.user
+ ).exists():
# you already boosted that.
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
models.Boost.objects.create(
boosted_status=status,
privacy=status.privacy,
user=request.user,
)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Unboost(View):
- ''' boost a status '''
+ """ boost a status """
+
def post(self, request, status_id):
- ''' boost a status '''
+ """ boost a status """
status = models.Status.objects.get(id=status_id)
boost = models.Boost.objects.filter(
boosted_status=status, user=request.user
).first()
boost.delete()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
diff --git a/bookwyrm/views/invite.py b/bookwyrm/views/invite.py
index 6b3611fc..1f1ccaf1 100644
--- a/bookwyrm/views/invite.py
+++ b/bookwyrm/views/invite.py
@@ -1,4 +1,4 @@
-''' invites when registration is closed '''
+""" invites when registration is closed """
from django.contrib.auth.decorators import login_required, permission_required
from django.core.paginator import Paginator
from django.http import HttpResponseBadRequest
@@ -12,32 +12,36 @@ from bookwyrm.settings import PAGE_LENGTH
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
@method_decorator(
- permission_required('bookwyrm.create_invites', raise_exception=True),
- name='dispatch')
+ permission_required("bookwyrm.create_invites", raise_exception=True),
+ name="dispatch",
+)
class ManageInvites(View):
- ''' create invites '''
+ """ create invites """
+
def get(self, request):
- ''' invite management page '''
+ """ invite management page """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
- paginated = Paginator(models.SiteInvite.objects.filter(
- user=request.user
- ).order_by('-created_date'), PAGE_LENGTH)
+ paginated = Paginator(
+ models.SiteInvite.objects.filter(user=request.user).order_by(
+ "-created_date"
+ ),
+ PAGE_LENGTH,
+ )
data = {
- 'title': 'Invitations',
- 'invites': paginated.page(page),
- 'form': forms.CreateInviteForm(),
+ "invites": paginated.page(page),
+ "form": forms.CreateInviteForm(),
}
- return TemplateResponse(request, 'settings/manage_invites.html', data)
+ return TemplateResponse(request, "settings/manage_invites.html", data)
def post(self, request):
- ''' creates an invite database entry '''
+ """ creates an invite database entry """
form = forms.CreateInviteForm(request.POST)
if not form.is_valid():
return HttpResponseBadRequest("ERRORS : %s" % (form.errors,))
@@ -46,31 +50,30 @@ class ManageInvites(View):
invite.user = request.user
invite.save()
- paginated = Paginator(models.SiteInvite.objects.filter(
- user=request.user
- ).order_by('-created_date'), PAGE_LENGTH)
- data = {
- 'title': 'Invitations',
- 'invites': paginated.page(1),
- 'form': form
- }
- return TemplateResponse(request, 'settings/manage_invites.html', data)
+ paginated = Paginator(
+ models.SiteInvite.objects.filter(user=request.user).order_by(
+ "-created_date"
+ ),
+ PAGE_LENGTH,
+ )
+ data = {"invites": paginated.page(1), "form": form}
+ return TemplateResponse(request, "settings/manage_invites.html", data)
class Invite(View):
- ''' use an invite to register '''
+ """ use an invite to register """
+
def get(self, request, code):
- ''' endpoint for using an invites '''
+ """ endpoint for using an invites """
if request.user.is_authenticated:
- return redirect('/')
+ return redirect("/")
invite = get_object_or_404(models.SiteInvite, code=code)
data = {
- 'title': 'Join',
- 'register_form': forms.RegisterForm(),
- 'invite': invite,
- 'valid': invite.valid() if invite else True,
+ "register_form": forms.RegisterForm(),
+ "invite": invite,
+ "valid": invite.valid() if invite else True,
}
- return TemplateResponse(request, 'invite.html', data)
+ return TemplateResponse(request, "invite.html", data)
# post handling is in views.authentication.Register
diff --git a/bookwyrm/views/isbn.py b/bookwyrm/views/isbn.py
new file mode 100644
index 00000000..b7ba02dd
--- /dev/null
+++ b/bookwyrm/views/isbn.py
@@ -0,0 +1,30 @@
+""" isbn search view """
+from django.http import HttpResponseNotFound
+from django.http import JsonResponse
+from django.shortcuts import get_object_or_404, redirect
+from django.template.response import TemplateResponse
+from django.utils.decorators import method_decorator
+from django.views import View
+from django.views.decorators.http import require_POST
+
+from bookwyrm import forms, models
+from bookwyrm.connectors import connector_manager
+from .helpers import is_api_request
+
+# pylint: disable= no-self-use
+class Isbn(View):
+ """ search a book by isbn """
+
+ def get(self, request, isbn):
+ """ info about a book """
+ book_results = connector_manager.isbn_local_search(isbn)
+
+ if is_api_request(request):
+ return JsonResponse([r.json() for r in book_results], safe=False)
+
+ data = {
+ "title": "ISBN Search Results",
+ "results": book_results,
+ "query": isbn,
+ }
+ return TemplateResponse(request, "isbn_search_results.html", data)
diff --git a/bookwyrm/views/landing.py b/bookwyrm/views/landing.py
index 2774e742..2c4a5147 100644
--- a/bookwyrm/views/landing.py
+++ b/bookwyrm/views/landing.py
@@ -1,4 +1,4 @@
-''' non-interactive pages '''
+""" non-interactive pages """
from django.db.models import Max
from django.template.response import TemplateResponse
from django.views import View
@@ -9,42 +9,44 @@ from .feed import Feed
# pylint: disable= no-self-use
class About(View):
- ''' create invites '''
+ """ create invites """
+
def get(self, request):
- ''' more information about the instance '''
- data = {
- 'title': 'About',
- }
- return TemplateResponse(request, 'discover/about.html', data)
+ """ more information about the instance """
+ return TemplateResponse(request, "discover/about.html")
+
class Home(View):
- ''' discover page or home feed depending on auth '''
+ """ discover page or home feed depending on auth """
+
def get(self, request):
- ''' this is the same as the feed on the home tab '''
+ """ this is the same as the feed on the home tab """
if request.user.is_authenticated:
feed_view = Feed.as_view()
- return feed_view(request, 'home')
+ return feed_view(request, "home")
discover_view = Discover.as_view()
return discover_view(request)
+
class Discover(View):
- ''' preview of recently reviewed books '''
+ """ preview of recently reviewed books """
+
def get(self, request):
- ''' tiled book activity page '''
- books = models.Edition.objects.filter(
- review__published_date__isnull=False,
- review__deleted=False,
- review__user__local=True,
- review__privacy__in=['public', 'unlisted'],
- ).exclude(
- cover__exact=''
- ).annotate(
- Max('review__published_date')
- ).order_by('-review__published_date__max')[:6]
+ """ tiled book activity page """
+ books = (
+ models.Edition.objects.filter(
+ review__published_date__isnull=False,
+ review__deleted=False,
+ review__user__local=True,
+ review__privacy__in=["public", "unlisted"],
+ )
+ .exclude(cover__exact="")
+ .annotate(Max("review__published_date"))
+ .order_by("-review__published_date__max")[:6]
+ )
data = {
- 'title': 'Discover',
- 'register_form': forms.RegisterForm(),
- 'books': list(set(books)),
+ "register_form": forms.RegisterForm(),
+ "books": list(set(books)),
}
- return TemplateResponse(request, 'discover/discover.html', data)
+ return TemplateResponse(request, "discover/discover.html", data)
diff --git a/bookwyrm/views/list.py b/bookwyrm/views/list.py
index e7b70a28..ba3200d1 100644
--- a/bookwyrm/views/list.py
+++ b/bookwyrm/views/list.py
@@ -1,4 +1,4 @@
-''' book list views'''
+""" book list views"""
from django.contrib.auth.decorators import login_required
from django.core.paginator import Paginator
from django.db import IntegrityError
@@ -18,52 +18,57 @@ from .helpers import get_user_from_username
# pylint: disable=no-self-use
class Lists(View):
- ''' book list page '''
+ """ book list page """
+
def get(self, request):
- ''' display a book list '''
+ """ display a book list """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
user = request.user if request.user.is_authenticated else None
# hide lists with no approved books
- lists = models.List.objects.filter(
- ~Q(user=user),
- ).annotate(
- item_count=Count('listitem', filter=Q(listitem__approved=True))
- ).filter(
- item_count__gt=0
- ).distinct().all()
+ lists = (
+ models.List.objects.filter(
+ ~Q(user=user),
+ )
+ .annotate(item_count=Count("listitem", filter=Q(listitem__approved=True)))
+ .filter(item_count__gt=0)
+ .distinct()
+ .all()
+ )
lists = privacy_filter(
- request.user, lists, privacy_levels=['public', 'followers'])
+ request.user, lists, privacy_levels=["public", "followers"]
+ )
paginated = Paginator(lists, 12)
data = {
- 'title': 'Lists',
- 'lists': paginated.page(page),
- 'list_form': forms.ListForm(),
- 'path': '/list',
+ "lists": paginated.page(page),
+ "list_form": forms.ListForm(),
+ "path": "/list",
}
- return TemplateResponse(request, 'lists/lists.html', data)
+ return TemplateResponse(request, "lists/lists.html", data)
- @method_decorator(login_required, name='dispatch')
+ @method_decorator(login_required, name="dispatch")
# pylint: disable=unused-argument
def post(self, request):
- ''' create a book_list '''
+ """ create a book_list """
form = forms.ListForm(request.POST)
if not form.is_valid():
- return redirect('lists')
+ return redirect("lists")
book_list = form.save()
return redirect(book_list.local_path)
+
class UserLists(View):
- ''' a user's book list page '''
+ """ a user's book list page """
+
def get(self, request, username):
- ''' display a book list '''
+ """ display a book list """
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
user = get_user_from_username(request.user, username)
@@ -72,20 +77,20 @@ class UserLists(View):
paginated = Paginator(lists, 12)
data = {
- 'title': '%s: Lists' % user.name,
- 'user': user,
- 'is_self': request.user.id == user.id,
- 'lists': paginated.page(page),
- 'list_form': forms.ListForm(),
- 'path': user.local_path + '/lists',
+ "user": user,
+ "is_self": request.user.id == user.id,
+ "lists": paginated.page(page),
+ "list_form": forms.ListForm(),
+ "path": user.local_path + "/lists",
}
- return TemplateResponse(request, 'user/lists.html', data)
+ return TemplateResponse(request, "user/lists.html", data)
class List(View):
- ''' book list page '''
+ """ book list page """
+
def get(self, request, list_id):
- ''' display a book list '''
+ """ display a book list """
book_list = get_object_or_404(models.List, id=list_id)
if not object_visible_to_user(request.user, book_list):
return HttpResponseNotFound()
@@ -93,7 +98,7 @@ class List(View):
if is_api_request(request):
return ActivitypubResponse(book_list.to_activity(**request.GET))
- query = request.GET.get('q')
+ query = request.GET.get("q")
suggestions = None
if query and request.user.is_authenticated:
# search for books
@@ -106,91 +111,85 @@ class List(View):
suggestions = [s.book for s in suggestions[:5]]
if len(suggestions) < 5:
suggestions += [
- s.default_edition for s in \
- models.Work.objects.filter(
- ~Q(editions__in=book_list.books.all()),
- ).order_by('-updated_date')
- ][:5 - len(suggestions)]
-
+ s.default_edition
+ for s in models.Work.objects.filter(
+ ~Q(editions__in=book_list.books.all()),
+ ).order_by("-updated_date")
+ ][: 5 - len(suggestions)]
data = {
- 'title': '%s | Lists' % book_list.name,
- 'list': book_list,
- 'items': book_list.listitem_set.filter(approved=True),
- 'pending_count': book_list.listitem_set.filter(
- approved=False).count(),
- 'suggested_books': suggestions,
- 'list_form': forms.ListForm(instance=book_list),
- 'query': query or ''
+ "list": book_list,
+ "items": book_list.listitem_set.filter(approved=True),
+ "pending_count": book_list.listitem_set.filter(approved=False).count(),
+ "suggested_books": suggestions,
+ "list_form": forms.ListForm(instance=book_list),
+ "query": query or "",
}
- return TemplateResponse(request, 'lists/list.html', data)
+ return TemplateResponse(request, "lists/list.html", data)
-
- @method_decorator(login_required, name='dispatch')
+ @method_decorator(login_required, name="dispatch")
# pylint: disable=unused-argument
def post(self, request, list_id):
- ''' edit a list '''
+ """ edit a list """
book_list = get_object_or_404(models.List, id=list_id)
form = forms.ListForm(request.POST, instance=book_list)
if not form.is_valid():
- return redirect('list', book_list.id)
+ return redirect("list", book_list.id)
book_list = form.save()
return redirect(book_list.local_path)
class Curate(View):
- ''' approve or discard list suggestsions '''
- @method_decorator(login_required, name='dispatch')
+ """ approve or discard list suggestsions """
+
+ @method_decorator(login_required, name="dispatch")
def get(self, request, list_id):
- ''' display a pending list '''
+ """ display a pending list """
book_list = get_object_or_404(models.List, id=list_id)
if not book_list.user == request.user:
# only the creater can curate the list
return HttpResponseNotFound()
data = {
- 'title': 'Curate "%s" | Lists' % book_list.name,
- 'list': book_list,
- 'pending': book_list.listitem_set.filter(approved=False),
- 'list_form': forms.ListForm(instance=book_list),
+ "list": book_list,
+ "pending": book_list.listitem_set.filter(approved=False),
+ "list_form": forms.ListForm(instance=book_list),
}
- return TemplateResponse(request, 'lists/curate.html', data)
+ return TemplateResponse(request, "lists/curate.html", data)
-
- @method_decorator(login_required, name='dispatch')
+ @method_decorator(login_required, name="dispatch")
# pylint: disable=unused-argument
def post(self, request, list_id):
- ''' edit a book_list '''
+ """ edit a book_list """
book_list = get_object_or_404(models.List, id=list_id)
- suggestion = get_object_or_404(
- models.ListItem, id=request.POST.get('item'))
- approved = request.POST.get('approved') == 'true'
+ suggestion = get_object_or_404(models.ListItem, id=request.POST.get("item"))
+ approved = request.POST.get("approved") == "true"
if approved:
suggestion.approved = True
suggestion.save()
else:
suggestion.delete()
- return redirect('list-curate', book_list.id)
+ return redirect("list-curate", book_list.id)
@require_POST
def add_book(request, list_id):
- ''' put a book on a list '''
+ """ put a book on a list """
book_list = get_object_or_404(models.List, id=list_id)
if not object_visible_to_user(request.user, book_list):
return HttpResponseNotFound()
- book = get_object_or_404(models.Edition, id=request.POST.get('book'))
+ book = get_object_or_404(models.Edition, id=request.POST.get("book"))
# do you have permission to add to the list?
try:
- if request.user == book_list.user or book_list.curation == 'open':
+ if request.user == book_list.user or book_list.curation == "open":
# go ahead and add it
models.ListItem.objects.create(
book=book,
book_list=book_list,
user=request.user,
)
- elif book_list.curation == 'curated':
+ elif book_list.curation == "curated":
# make a pending entry
models.ListItem.objects.create(
approved=False,
@@ -205,17 +204,17 @@ def add_book(request, list_id):
# if the book is already on the list, don't flip out
pass
- return redirect('list', list_id)
+ return redirect("list", list_id)
@require_POST
def remove_book(request, list_id):
- ''' put a book on a list '''
+ """ put a book on a list """
book_list = get_object_or_404(models.List, id=list_id)
- item = get_object_or_404(models.ListItem, id=request.POST.get('item'))
+ item = get_object_or_404(models.ListItem, id=request.POST.get("item"))
if not book_list.user == request.user and not item.user == request.user:
return HttpResponseNotFound()
item.delete()
- return redirect('list', list_id)
+ return redirect("list", list_id)
diff --git a/bookwyrm/views/notifications.py b/bookwyrm/views/notifications.py
index 7d6a3149..7a62ec01 100644
--- a/bookwyrm/views/notifications.py
+++ b/bookwyrm/views/notifications.py
@@ -1,4 +1,4 @@
-''' non-interactive pages '''
+""" non-interactive pages """
from django.contrib.auth.decorators import login_required
from django.template.response import TemplateResponse
from django.utils.decorators import method_decorator
@@ -7,23 +7,22 @@ from django.views import View
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Notifications(View):
- ''' notifications view '''
+ """ notifications view """
+
def get(self, request):
- ''' people are interacting with you, get hyped '''
- notifications = request.user.notification_set.all() \
- .order_by('-created_date')
+ """ people are interacting with you, get hyped """
+ notifications = request.user.notification_set.all().order_by("-created_date")
unread = [n.id for n in notifications.filter(read=False)]
data = {
- 'title': 'Notifications',
- 'notifications': notifications,
- 'unread': unread,
+ "notifications": notifications,
+ "unread": unread,
}
notifications.update(read=True)
- return TemplateResponse(request, 'notifications.html', data)
+ return TemplateResponse(request, "notifications.html", data)
def post(self, request):
- ''' permanently delete notification for user '''
+ """ permanently delete notification for user """
request.user.notification_set.filter(read=True).delete()
- return redirect('/notifications')
+ return redirect("/notifications")
diff --git a/bookwyrm/views/outbox.py b/bookwyrm/views/outbox.py
index 5df9d199..ec6f5cd3 100644
--- a/bookwyrm/views/outbox.py
+++ b/bookwyrm/views/outbox.py
@@ -1,4 +1,4 @@
-''' the good stuff! the books! '''
+""" the good stuff! the books! """
from django.http import JsonResponse
from django.shortcuts import get_object_or_404
from django.views import View
@@ -9,11 +9,12 @@ from .helpers import is_bookwyrm_request
# pylint: disable= no-self-use
class Outbox(View):
- ''' outbox '''
+ """ outbox """
+
def get(self, request, username):
- ''' outbox for the requested user '''
+ """ outbox for the requested user """
user = get_object_or_404(models.User, localname=username)
- filter_type = request.GET.get('type')
+ filter_type = request.GET.get("type")
if filter_type not in models.status_models:
filter_type = None
@@ -23,5 +24,5 @@ class Outbox(View):
filter_type=filter_type,
pure=not is_bookwyrm_request(request)
),
- encoder=activitypub.ActivityEncoder
+ encoder=activitypub.ActivityEncoder,
)
diff --git a/bookwyrm/views/password.py b/bookwyrm/views/password.py
index 6602a270..e853d16b 100644
--- a/bookwyrm/views/password.py
+++ b/bookwyrm/views/password.py
@@ -1,4 +1,4 @@
-''' class views for password management '''
+""" class views for password management """
from django.contrib.auth import login
from django.contrib.auth.decorators import login_required
from django.core.exceptions import PermissionDenied
@@ -13,22 +13,22 @@ from bookwyrm.emailing import password_reset_email
# pylint: disable= no-self-use
class PasswordResetRequest(View):
- ''' forgot password flow '''
+ """ forgot password flow """
+
def get(self, request):
- ''' password reset page '''
+ """ password reset page """
return TemplateResponse(
request,
- 'password_reset_request.html',
- {'title': 'Reset Password'}
+ "password_reset_request.html",
)
def post(self, request):
- ''' create a password reset token '''
- email = request.POST.get('email')
+ """ create a password reset token """
+ email = request.POST.get("email")
try:
user = models.User.objects.get(email=email)
except models.User.DoesNotExist:
- return redirect('/password-reset')
+ return redirect("/password-reset")
# remove any existing password reset cods for this user
models.PasswordReset.objects.filter(user=user).all().delete()
@@ -36,16 +36,17 @@ class PasswordResetRequest(View):
# create a new reset code
code = models.PasswordReset.objects.create(user=user)
password_reset_email(code)
- data = {'message': 'Password reset link sent to %s' % email}
- return TemplateResponse(request, 'password_reset_request.html', data)
+ data = {"message": "Password reset link sent to %s" % email}
+ return TemplateResponse(request, "password_reset_request.html", data)
class PasswordReset(View):
- ''' set new password '''
+ """ set new password """
+
def get(self, request, code):
- ''' endpoint for sending invites '''
+ """ endpoint for sending invites """
if request.user.is_authenticated:
- return redirect('/')
+ return redirect("/")
try:
reset_code = models.PasswordReset.objects.get(code=code)
if not reset_code.valid():
@@ -53,57 +54,48 @@ class PasswordReset(View):
except models.PasswordReset.DoesNotExist:
raise PermissionDenied
- return TemplateResponse(
- request,
- 'password_reset.html',
- {'title': 'Reset Password', 'code': reset_code.code}
- )
+ return TemplateResponse(request, "password_reset.html")
def post(self, request, code):
- ''' allow a user to change their password through an emailed token '''
+ """ allow a user to change their password through an emailed token """
try:
- reset_code = models.PasswordReset.objects.get(
- code=code
- )
+ reset_code = models.PasswordReset.objects.get(code=code)
except models.PasswordReset.DoesNotExist:
- data = {'errors': ['Invalid password reset link']}
- return TemplateResponse(request, 'password_reset.html', data)
+ data = {"errors": ["Invalid password reset link"]}
+ return TemplateResponse(request, "password_reset.html", data)
user = reset_code.user
- new_password = request.POST.get('password')
- confirm_password = request.POST.get('confirm-password')
+ new_password = request.POST.get("password")
+ confirm_password = request.POST.get("confirm-password")
if new_password != confirm_password:
- data = {'errors': ['Passwords do not match']}
- return TemplateResponse(request, 'password_reset.html', data)
+ data = {"errors": ["Passwords do not match"]}
+ return TemplateResponse(request, "password_reset.html", data)
user.set_password(new_password)
user.save(broadcast=False)
login(request, user)
reset_code.delete()
- return redirect('/')
+ return redirect("/")
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class ChangePassword(View):
- ''' change password as logged in user '''
+ """ change password as logged in user """
+
def get(self, request):
- ''' change password page '''
- data = {
- 'title': 'Change Password',
- 'user': request.user,
- }
- return TemplateResponse(
- request, 'preferences/change_password.html', data)
+ """ change password page """
+ data = {"user": request.user}
+ return TemplateResponse(request, "preferences/change_password.html", data)
def post(self, request):
- ''' allow a user to change their password '''
- new_password = request.POST.get('password')
- confirm_password = request.POST.get('confirm-password')
+ """ allow a user to change their password """
+ new_password = request.POST.get("password")
+ confirm_password = request.POST.get("confirm-password")
if new_password != confirm_password:
- return redirect('preferences/password')
+ return redirect("preferences/password")
request.user.set_password(new_password)
request.user.save(broadcast=False)
diff --git a/bookwyrm/views/reading.py b/bookwyrm/views/reading.py
index 6b2bf242..fb63f2a3 100644
--- a/bookwyrm/views/reading.py
+++ b/bookwyrm/views/reading.py
@@ -1,4 +1,4 @@
-''' the good stuff! the books! '''
+""" the good stuff! the books! """
import dateutil.parser
from dateutil.parser import ParserError
@@ -17,12 +17,9 @@ from .shelf import handle_unshelve
@login_required
@require_POST
def start_reading(request, book_id):
- ''' begin reading a book '''
+ """ begin reading a book """
book = get_edition(book_id)
- shelf = models.Shelf.objects.filter(
- identifier='reading',
- user=request.user
- ).first()
+ shelf = models.Shelf.objects.filter(identifier="reading", user=request.user).first()
# create a readthrough
readthrough = update_readthrough(request, book=book)
@@ -33,36 +30,29 @@ def start_reading(request, book_id):
readthrough.create_update()
# shelve the book
- if request.POST.get('reshelve', True):
+ if request.POST.get("reshelve", True):
try:
- current_shelf = models.Shelf.objects.get(
- user=request.user,
- edition=book
- )
+ current_shelf = models.Shelf.objects.get(user=request.user, edition=book)
handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
- models.ShelfBook.objects.create(
- book=book, shelf=shelf, user=request.user)
+ models.ShelfBook.objects.create(book=book, shelf=shelf, user=request.user)
# post about it (if you want)
- if request.POST.get('post-status'):
- privacy = request.POST.get('privacy')
+ if request.POST.get("post-status"):
+ privacy = request.POST.get("privacy")
handle_reading_status(request.user, shelf, book, privacy)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
@login_required
@require_POST
def finish_reading(request, book_id):
- ''' a user completed a book, yay '''
+ """ a user completed a book, yay """
book = get_edition(book_id)
- shelf = models.Shelf.objects.filter(
- identifier='read',
- user=request.user
- ).first()
+ shelf = models.Shelf.objects.filter(identifier="read", user=request.user).first()
# update or create a readthrough
readthrough = update_readthrough(request, book=book)
@@ -70,31 +60,27 @@ def finish_reading(request, book_id):
readthrough.save()
# shelve the book
- if request.POST.get('reshelve', True):
+ if request.POST.get("reshelve", True):
try:
- current_shelf = models.Shelf.objects.get(
- user=request.user,
- edition=book
- )
+ current_shelf = models.Shelf.objects.get(user=request.user, edition=book)
handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
- models.ShelfBook.objects.create(
- book=book, shelf=shelf, user=request.user)
+ models.ShelfBook.objects.create(book=book, shelf=shelf, user=request.user)
# post about it (if you want)
- if request.POST.get('post-status'):
- privacy = request.POST.get('privacy')
+ if request.POST.get("post-status"):
+ privacy = request.POST.get("privacy")
handle_reading_status(request.user, shelf, book, privacy)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
@login_required
@require_POST
def edit_readthrough(request):
- ''' can't use the form because the dates are too finnicky '''
+ """ can't use the form because the dates are too finnicky """
readthrough = update_readthrough(request, create=False)
if not readthrough:
return HttpResponseNotFound()
@@ -108,40 +94,39 @@ def edit_readthrough(request):
# use default now for date field
readthrough.create_update()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
@login_required
@require_POST
def delete_readthrough(request):
- ''' remove a readthrough '''
- readthrough = get_object_or_404(
- models.ReadThrough, id=request.POST.get('id'))
+ """ remove a readthrough """
+ readthrough = get_object_or_404(models.ReadThrough, id=request.POST.get("id"))
# don't let people edit other people's data
if request.user != readthrough.user:
return HttpResponseBadRequest()
readthrough.delete()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
@login_required
@require_POST
def create_readthrough(request):
- ''' can't use the form because the dates are too finnicky '''
- book = get_object_or_404(models.Edition, id=request.POST.get('book'))
+ """ can't use the form because the dates are too finnicky """
+ book = get_object_or_404(models.Edition, id=request.POST.get("book"))
readthrough = update_readthrough(request, create=True, book=book)
if not readthrough:
return redirect(book.local_path)
readthrough.save()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
def update_readthrough(request, book=None, create=True):
- ''' updates but does not save dates on a readthrough '''
+ """ updates but does not save dates on a readthrough """
try:
- read_id = request.POST.get('id')
+ read_id = request.POST.get("id")
if not read_id:
raise models.ReadThrough.DoesNotExist
readthrough = models.ReadThrough.objects.get(id=read_id)
@@ -153,7 +138,7 @@ def update_readthrough(request, book=None, create=True):
book=book,
)
- start_date = request.POST.get('start_date')
+ start_date = request.POST.get("start_date")
if start_date:
try:
start_date = timezone.make_aware(dateutil.parser.parse(start_date))
@@ -161,16 +146,15 @@ def update_readthrough(request, book=None, create=True):
except ParserError:
pass
- finish_date = request.POST.get('finish_date')
+ finish_date = request.POST.get("finish_date")
if finish_date:
try:
- finish_date = timezone.make_aware(
- dateutil.parser.parse(finish_date))
+ finish_date = timezone.make_aware(dateutil.parser.parse(finish_date))
readthrough.finish_date = finish_date
except ParserError:
pass
- progress = request.POST.get('progress')
+ progress = request.POST.get("progress")
if progress:
try:
progress = int(progress)
@@ -178,7 +162,7 @@ def update_readthrough(request, book=None, create=True):
except ValueError:
pass
- progress_mode = request.POST.get('progress_mode')
+ progress_mode = request.POST.get("progress_mode")
if progress_mode:
try:
progress_mode = models.ProgressMode(progress_mode)
@@ -191,15 +175,16 @@ def update_readthrough(request, book=None, create=True):
return readthrough
+
@login_required
@require_POST
def delete_progressupdate(request):
- ''' remove a progress update '''
- update = get_object_or_404(models.ProgressUpdate, id=request.POST.get('id'))
+ """ remove a progress update """
+ update = get_object_or_404(models.ProgressUpdate, id=request.POST.get("id"))
# don't let people edit other people's data
if request.user != update.user:
return HttpResponseBadRequest()
update.delete()
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
diff --git a/bookwyrm/views/rss_feed.py b/bookwyrm/views/rss_feed.py
index d24b636e..57821af4 100644
--- a/bookwyrm/views/rss_feed.py
+++ b/bookwyrm/views/rss_feed.py
@@ -1,38 +1,35 @@
-''' serialize user's posts in rss feed '''
+""" serialize user's posts in rss feed """
from django.contrib.syndication.views import Feed
from .helpers import get_activity_feed, get_user_from_username
# pylint: disable=no-self-use, unused-argument
class RssFeed(Feed):
- ''' serialize user's posts in rss feed '''
- description_template = 'snippets/rss_content.html'
- title_template = 'snippets/rss_title.html'
+ """ serialize user's posts in rss feed """
+
+ description_template = "snippets/rss_content.html"
+ title_template = "snippets/rss_title.html"
def get_object(self, request, username):
- ''' the user who's posts get serialized '''
+ """ the user who's posts get serialized """
return get_user_from_username(request.user, username)
-
def link(self, obj):
- ''' link to the user's profile '''
+ """ link to the user's profile """
return obj.local_path
-
def title(self, obj):
- ''' title of the rss feed entry '''
- return f'Status updates from {obj.display_name}'
-
+ """ title of the rss feed entry """
+ return f"Status updates from {obj.display_name}"
def items(self, obj):
- ''' the user's activity feed '''
+ """ the user's activity feed """
return get_activity_feed(
obj,
- privacy=['public', 'unlisted'],
- queryset=obj.status_set.select_subclasses()
+ privacy=["public", "unlisted"],
+ queryset=obj.status_set.select_subclasses(),
)
-
def item_link(self, item):
- ''' link to the status '''
+ """ link to the status """
return item.local_path
diff --git a/bookwyrm/views/search.py b/bookwyrm/views/search.py
index 8acb2836..80969d31 100644
--- a/bookwyrm/views/search.py
+++ b/bookwyrm/views/search.py
@@ -1,4 +1,4 @@
-''' search views'''
+""" search views"""
import re
from django.contrib.postgres.search import TrigramSimilarity
@@ -16,52 +16,63 @@ from .helpers import handle_remote_webfinger
# pylint: disable= no-self-use
class Search(View):
- ''' search users or books '''
+ """ search users or books """
+
def get(self, request):
- ''' that search bar up top '''
- query = request.GET.get('q')
- min_confidence = request.GET.get('min_confidence', 0.1)
+ """ that search bar up top """
+ query = request.GET.get("q")
+ min_confidence = request.GET.get("min_confidence", 0.1)
if is_api_request(request):
# only return local book results via json so we don't cascade
book_results = connector_manager.local_search(
- query, min_confidence=min_confidence)
+ query, min_confidence=min_confidence
+ )
return JsonResponse([r.json() for r in book_results], safe=False)
# use webfinger for mastodon style account@domain.com username
- if re.match(r'\B%s' % regex.full_username, query):
+ if re.match(r"\B%s" % regex.full_username, query):
handle_remote_webfinger(query)
# do a user search
- user_results = models.User.viewer_aware_objects(request.user).annotate(
- similarity=Greatest(
- TrigramSimilarity('username', query),
- TrigramSimilarity('localname', query),
+ user_results = (
+ models.User.viewer_aware_objects(request.user)
+ .annotate(
+ similarity=Greatest(
+ TrigramSimilarity("username", query),
+ TrigramSimilarity("localname", query),
+ )
)
- ).filter(
- similarity__gt=0.5,
- ).order_by('-similarity')[:10]
+ .filter(
+ similarity__gt=0.5,
+ )
+ .order_by("-similarity")[:10]
+ )
# any relevent lists?
- list_results = privacy_filter(
- request.user, models.List.objects,
- privacy_levels=['public', 'followers']
- ).annotate(
- similarity=Greatest(
- TrigramSimilarity('name', query),
- TrigramSimilarity('description', query),
+ list_results = (
+ privacy_filter(
+ request.user,
+ models.List.objects,
+ privacy_levels=["public", "followers"],
)
- ).filter(
- similarity__gt=0.1,
- ).order_by('-similarity')[:10]
+ .annotate(
+ similarity=Greatest(
+ TrigramSimilarity("name", query),
+ TrigramSimilarity("description", query),
+ )
+ )
+ .filter(
+ similarity__gt=0.1,
+ )
+ .order_by("-similarity")[:10]
+ )
- book_results = connector_manager.search(
- query, min_confidence=min_confidence)
+ book_results = connector_manager.search(query, min_confidence=min_confidence)
data = {
- 'title': 'Search Results',
- 'book_results': book_results,
- 'user_results': user_results,
- 'list_results': list_results,
- 'query': query,
+ "book_results": book_results,
+ "user_results": user_results,
+ "list_results": list_results,
+ "query": query,
}
- return TemplateResponse(request, 'search_results.html', data)
+ return TemplateResponse(request, "search_results.html", data)
diff --git a/bookwyrm/views/shelf.py b/bookwyrm/views/shelf.py
index 70d3d1de..6256eac7 100644
--- a/bookwyrm/views/shelf.py
+++ b/bookwyrm/views/shelf.py
@@ -1,4 +1,4 @@
-''' shelf views'''
+""" shelf views"""
from django.contrib.auth.decorators import login_required
from django.http import HttpResponseBadRequest, HttpResponseNotFound
from django.shortcuts import get_object_or_404, redirect
@@ -15,9 +15,10 @@ from .helpers import handle_reading_status
# pylint: disable= no-self-use
class Shelf(View):
- ''' shelf page '''
+ """ shelf page """
+
def get(self, request, username, shelf_identifier):
- ''' display a shelf '''
+ """ display a shelf """
try:
user = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -34,39 +35,40 @@ class Shelf(View):
if not is_self:
follower = user.followers.filter(id=request.user.id).exists()
# make sure the user has permission to view the shelf
- if shelf.privacy == 'direct' or \
- (shelf.privacy == 'followers' and not follower):
+ if shelf.privacy == "direct" or (
+ shelf.privacy == "followers" and not follower
+ ):
return HttpResponseNotFound()
# only show other shelves that should be visible
if follower:
- shelves = shelves.filter(privacy__in=['public', 'followers'])
+ shelves = shelves.filter(privacy__in=["public", "followers"])
else:
- shelves = shelves.filter(privacy='public')
-
+ shelves = shelves.filter(privacy="public")
if is_api_request(request):
return ActivitypubResponse(shelf.to_activity(**request.GET))
- books = models.ShelfBook.objects.filter(
- user=user, shelf=shelf
- ).order_by('-updated_date').all()
+ books = (
+ models.ShelfBook.objects.filter(user=user, shelf=shelf)
+ .order_by("-updated_date")
+ .all()
+ )
data = {
- 'title': '%s\'s %s shelf' % (user.display_name, shelf.name),
- 'user': user,
- 'is_self': is_self,
- 'shelves': shelves.all(),
- 'shelf': shelf,
- 'books': [b.book for b in books],
+ "user": user,
+ "is_self": is_self,
+ "shelves": shelves.all(),
+ "shelf": shelf,
+ "books": [b.book for b in books],
}
- return TemplateResponse(request, 'user/shelf.html', data)
+ return TemplateResponse(request, "user/shelf.html", data)
- @method_decorator(login_required, name='dispatch')
+ @method_decorator(login_required, name="dispatch")
# pylint: disable=unused-argument
def post(self, request, username, shelf_identifier):
- ''' edit a shelf '''
+ """ edit a shelf """
try:
shelf = request.user.shelf_set.get(identifier=shelf_identifier)
except models.Shelf.DoesNotExist:
@@ -74,7 +76,7 @@ class Shelf(View):
if request.user != shelf.user:
return HttpResponseBadRequest()
- if not shelf.editable and request.POST.get('name') != shelf.name:
+ if not shelf.editable and request.POST.get("name") != shelf.name:
return HttpResponseBadRequest()
form = forms.ShelfForm(request.POST, instance=shelf)
@@ -85,88 +87,76 @@ class Shelf(View):
def user_shelves_page(request, username):
- ''' default shelf '''
+ """ default shelf """
return Shelf.as_view()(request, username, None)
@login_required
@require_POST
def create_shelf(request):
- ''' user generated shelves '''
+ """ user generated shelves """
form = forms.ShelfForm(request.POST)
if not form.is_valid():
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
shelf = form.save()
- return redirect('/user/%s/shelf/%s' % \
- (request.user.localname, shelf.identifier))
+ return redirect("/user/%s/shelf/%s" % (request.user.localname, shelf.identifier))
@login_required
@require_POST
def delete_shelf(request, shelf_id):
- ''' user generated shelves '''
+ """ user generated shelves """
shelf = get_object_or_404(models.Shelf, id=shelf_id)
if request.user != shelf.user or not shelf.editable:
return HttpResponseBadRequest()
shelf.delete()
- return redirect('/user/%s/shelves' % request.user.localname)
+ return redirect("/user/%s/shelves" % request.user.localname)
@login_required
@require_POST
def shelve(request):
- ''' put a on a user's shelf '''
- book = get_edition(request.POST.get('book'))
+ """ put a on a user's shelf """
+ book = get_edition(request.POST.get("book"))
desired_shelf = models.Shelf.objects.filter(
- identifier=request.POST.get('shelf'),
- user=request.user
+ identifier=request.POST.get("shelf"), user=request.user
).first()
if not desired_shelf:
return HttpResponseNotFound()
- if request.POST.get('reshelve', True):
+ if request.POST.get("reshelve", True):
try:
- current_shelf = models.Shelf.objects.get(
- user=request.user,
- edition=book
- )
+ current_shelf = models.Shelf.objects.get(user=request.user, edition=book)
handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
- models.ShelfBook.objects.create(
- book=book, shelf=desired_shelf, user=request.user)
+ models.ShelfBook.objects.create(book=book, shelf=desired_shelf, user=request.user)
# post about "want to read" shelves
- if desired_shelf.identifier == 'to-read' and \
- request.POST.get('post-status'):
- privacy = request.POST.get('privacy') or desired_shelf.privacy
- handle_reading_status(
- request.user,
- desired_shelf,
- book,
- privacy=privacy
- )
+ if desired_shelf.identifier == "to-read" and request.POST.get("post-status"):
+ privacy = request.POST.get("privacy") or desired_shelf.privacy
+ handle_reading_status(request.user, desired_shelf, book, privacy=privacy)
- return redirect('/')
+ return redirect("/")
@login_required
@require_POST
def unshelve(request):
- ''' put a on a user's shelf '''
- book = models.Edition.objects.get(id=request.POST['book'])
- current_shelf = models.Shelf.objects.get(id=request.POST['shelf'])
+ """ put a on a user's shelf """
+ book = models.Edition.objects.get(id=request.POST["book"])
+ current_shelf = models.Shelf.objects.get(id=request.POST["shelf"])
handle_unshelve(request.user, book, current_shelf)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
-#pylint: disable=unused-argument
+# pylint: disable=unused-argument
def handle_unshelve(user, book, shelf):
- ''' unshelve a book '''
+ """ unshelve a book """
row = models.ShelfBook.objects.get(book=book, shelf=shelf)
row.delete()
diff --git a/bookwyrm/views/site.py b/bookwyrm/views/site.py
index 0a5e270b..c40a9e76 100644
--- a/bookwyrm/views/site.py
+++ b/bookwyrm/views/site.py
@@ -1,4 +1,4 @@
-''' manage site settings '''
+""" manage site settings """
from django.contrib.auth.decorators import login_required, permission_required
from django.shortcuts import redirect
from django.template.response import TemplateResponse
@@ -9,32 +9,27 @@ from bookwyrm import forms, models
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
@method_decorator(
- permission_required(
- 'bookwyrm.edit_instance_settings', raise_exception=True),
- name='dispatch')
+ permission_required("bookwyrm.edit_instance_settings", raise_exception=True),
+ name="dispatch",
+)
class Site(View):
- ''' manage things like the instance name '''
+ """ manage things like the instance name """
+
def get(self, request):
- ''' edit form '''
+ """ edit form """
site = models.SiteSettings.objects.get()
- data = {
- 'title': 'Site Settings',
- 'site_form': forms.SiteForm(instance=site)
- }
- return TemplateResponse(request, 'settings/site.html', data)
+ data = {"site_form": forms.SiteForm(instance=site)}
+ return TemplateResponse(request, "settings/site.html", data)
def post(self, request):
- ''' edit the site settings '''
+ """ edit the site settings """
site = models.SiteSettings.objects.get()
form = forms.SiteForm(request.POST, instance=site)
if not form.is_valid():
- data = {
- 'title': 'Site Settings',
- 'site_form': form
- }
- return TemplateResponse(request, 'settings/site.html', data)
+ data = {"site_form": form}
+ return TemplateResponse(request, "settings/site.html", data)
form.save()
- return redirect('settings-site')
+ return redirect("settings-site")
diff --git a/bookwyrm/views/status.py b/bookwyrm/views/status.py
index 983fea38..8cc5df69 100644
--- a/bookwyrm/views/status.py
+++ b/bookwyrm/views/status.py
@@ -1,4 +1,4 @@
-''' what are we here for if not for posting '''
+""" what are we here for if not for posting """
import re
from django.contrib.auth.decorators import login_required
from django.http import HttpResponseBadRequest
@@ -16,19 +16,20 @@ from .helpers import handle_remote_webfinger
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class CreateStatus(View):
- ''' the view for *posting* '''
+ """ the view for *posting* """
+
def post(self, request, status_type):
- ''' create status of whatever type '''
+ """ create status of whatever type """
status_type = status_type[0].upper() + status_type[1:]
try:
- form = getattr(forms, '%sForm' % status_type)(request.POST)
+ form = getattr(forms, "%sForm" % status_type)(request.POST)
except AttributeError:
return HttpResponseBadRequest()
if not form.is_valid():
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
status = form.save(commit=False)
if not status.sensitive and status.content_warning:
@@ -44,10 +45,10 @@ class CreateStatus(View):
# turn the mention into a link
content = re.sub(
- r'%s([^@]|$)' % mention_text,
- r'%s \g<1>' % \
- (mention_user.remote_id, mention_text),
- content)
+ r"%s([^@]|$)" % mention_text,
+ r'%s \g<1>' % (mention_user.remote_id, mention_text),
+ content,
+ )
# add reply parent to mentions
if status.reply_parent:
status.mention_users.add(status.reply_parent.user)
@@ -59,17 +60,18 @@ class CreateStatus(View):
if not isinstance(status, models.GeneratedNote) and content:
status.content = to_markdown(content)
# do apply formatting to quotes
- if hasattr(status, 'quote'):
+ if hasattr(status, "quote"):
status.quote = to_markdown(status.quote)
status.save(created=True)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
class DeleteStatus(View):
- ''' tombstone that bad boy '''
+ """ tombstone that bad boy """
+
def post(self, request, status_id):
- ''' delete and tombstone a status '''
+ """ delete and tombstone a status """
status = get_object_or_404(models.Status, id=status_id)
# don't let people delete other people's statuses
@@ -78,18 +80,19 @@ class DeleteStatus(View):
# perform deletion
delete_status(status)
- return redirect(request.headers.get('Referer', '/'))
+ return redirect(request.headers.get("Referer", "/"))
+
def find_mentions(content):
- ''' detect @mentions in raw status content '''
+ """ detect @mentions in raw status content """
if not content:
return
for match in re.finditer(regex.strict_username, content):
- username = match.group().strip().split('@')[1:]
+ username = match.group().strip().split("@")[1:]
if len(username) == 1:
# this looks like a local user (@user), fill in the domain
username.append(DOMAIN)
- username = '@'.join(username)
+ username = "@".join(username)
mention_user = handle_remote_webfinger(username)
if not mention_user:
@@ -99,15 +102,16 @@ def find_mentions(content):
def format_links(content):
- ''' detect and format links '''
+ """ detect and format links """
return re.sub(
- r'([^(href=")]|^|\()(https?:\/\/(%s([\w\.\-_\/+&\?=:;,])*))' % \
- regex.domain,
+ r'([^(href=")]|^|\()(https?:\/\/(%s([\w\.\-_\/+&\?=:;,])*))' % regex.domain,
r'\g<1>\g<3> ',
- content)
+ content,
+ )
+
def to_markdown(content):
- ''' catch links and convert to markdown '''
+ """ catch links and convert to markdown """
content = markdown(content)
content = format_links(content)
# sanitize resulting html
diff --git a/bookwyrm/views/tag.py b/bookwyrm/views/tag.py
index 502f5ea5..a6bdf05a 100644
--- a/bookwyrm/views/tag.py
+++ b/bookwyrm/views/tag.py
@@ -1,4 +1,4 @@
-''' tagging views'''
+""" tagging views"""
from django.contrib.auth.decorators import login_required
from django.shortcuts import get_object_or_404, redirect
from django.template.response import TemplateResponse
@@ -12,35 +12,35 @@ from .helpers import is_api_request
# pylint: disable= no-self-use
class Tag(View):
- ''' tag page '''
+ """ tag page """
+
def get(self, request, tag_id):
- ''' see books related to a tag '''
+ """ see books related to a tag """
tag_obj = get_object_or_404(models.Tag, identifier=tag_id)
if is_api_request(request):
- return ActivitypubResponse(
- tag_obj.to_activity(**request.GET))
+ return ActivitypubResponse(tag_obj.to_activity(**request.GET))
books = models.Edition.objects.filter(
usertag__tag__identifier=tag_id
).distinct()
data = {
- 'title': tag_obj.name,
- 'books': books,
- 'tag': tag_obj,
+ "books": books,
+ "tag": tag_obj,
}
- return TemplateResponse(request, 'tag.html', data)
+ return TemplateResponse(request, "tag.html", data)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class AddTag(View):
- ''' add a tag to a book '''
+ """ add a tag to a book """
+
def post(self, request):
- ''' tag a book '''
+ """ tag a book """
# I'm not using a form here because sometimes "name" is sent as a hidden
# field which doesn't validate
- name = request.POST.get('name')
- book_id = request.POST.get('book')
+ name = request.POST.get("name")
+ book_id = request.POST.get("book")
book = get_object_or_404(models.Edition, id=book_id)
tag_obj, _ = models.Tag.objects.get_or_create(
name=name,
@@ -51,21 +51,23 @@ class AddTag(View):
tag=tag_obj,
)
- return redirect('/book/%s' % book_id)
+ return redirect("/book/%s" % book_id)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class RemoveTag(View):
- ''' remove a user's tag from a book '''
+ """ remove a user's tag from a book """
+
def post(self, request):
- ''' untag a book '''
- name = request.POST.get('name')
+ """ untag a book """
+ name = request.POST.get("name")
tag_obj = get_object_or_404(models.Tag, name=name)
- book_id = request.POST.get('book')
+ book_id = request.POST.get("book")
book = get_object_or_404(models.Edition, id=book_id)
user_tag = get_object_or_404(
- models.UserTag, tag=tag_obj, book=book, user=request.user)
+ models.UserTag, tag=tag_obj, book=book, user=request.user
+ )
user_tag.delete()
- return redirect('/book/%s' % book_id)
+ return redirect("/book/%s" % book_id)
diff --git a/bookwyrm/views/updates.py b/bookwyrm/views/updates.py
index 233e5191..83b680c0 100644
--- a/bookwyrm/views/updates.py
+++ b/bookwyrm/views/updates.py
@@ -1,17 +1,20 @@
-''' endpoints for getting updates about activity '''
+""" endpoints for getting updates about activity """
from django.contrib.auth.decorators import login_required
from django.http import JsonResponse
from django.utils.decorators import method_decorator
from django.views import View
# pylint: disable= no-self-use
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class Updates(View):
- ''' so the app can poll '''
+ """ so the app can poll """
+
def get(self, request):
- ''' any notifications waiting? '''
- return JsonResponse({
- 'notifications': request.user.notification_set.filter(
- read=False
- ).count(),
- })
+ """ any notifications waiting? """
+ return JsonResponse(
+ {
+ "notifications": request.user.notification_set.filter(
+ read=False
+ ).count(),
+ }
+ )
diff --git a/bookwyrm/views/user.py b/bookwyrm/views/user.py
index a218375f..469a82d3 100644
--- a/bookwyrm/views/user.py
+++ b/bookwyrm/views/user.py
@@ -1,4 +1,4 @@
-''' non-interactive pages '''
+""" non-interactive pages """
from io import BytesIO
from uuid import uuid4
from PIL import Image
@@ -22,9 +22,10 @@ from .helpers import is_blocked, object_visible_to_user
# pylint: disable= no-self-use
class User(View):
- ''' user profile page '''
+ """ user profile page """
+
def get(self, request, username):
- ''' profile page for a user '''
+ """ profile page for a user """
try:
user = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -40,7 +41,7 @@ class User(View):
# otherwise we're at a UI view
try:
- page = int(request.GET.get('page', 1))
+ page = int(request.GET.get("page", 1))
except ValueError:
page = 1
@@ -52,19 +53,21 @@ class User(View):
if not is_self:
follower = user.followers.filter(id=request.user.id).exists()
if follower:
- shelves = shelves.filter(privacy__in=['public', 'followers'])
+ shelves = shelves.filter(privacy__in=["public", "followers"])
else:
- shelves = shelves.filter(privacy='public')
+ shelves = shelves.filter(privacy="public")
for user_shelf in shelves.all():
if not user_shelf.books.count():
continue
- shelf_preview.append({
- 'name': user_shelf.name,
- 'local_path': user_shelf.local_path,
- 'books': user_shelf.books.all()[:3],
- 'size': user_shelf.books.count(),
- })
+ shelf_preview.append(
+ {
+ "name": user_shelf.name,
+ "local_path": user_shelf.local_path,
+ "books": user_shelf.books.all()[:3],
+ "size": user_shelf.books.count(),
+ }
+ )
if len(shelf_preview) > 2:
break
@@ -75,25 +78,27 @@ class User(View):
)
paginated = Paginator(activities, PAGE_LENGTH)
goal = models.AnnualGoal.objects.filter(
- user=user, year=timezone.now().year).first()
+ user=user, year=timezone.now().year
+ ).first()
if not object_visible_to_user(request.user, goal):
goal = None
data = {
- 'title': user.name,
- 'user': user,
- 'is_self': is_self,
- 'shelves': shelf_preview,
- 'shelf_count': shelves.count(),
- 'activities': paginated.page(page),
- 'goal': goal,
+ "user": user,
+ "is_self": is_self,
+ "shelves": shelf_preview,
+ "shelf_count": shelves.count(),
+ "activities": paginated.page(page),
+ "goal": goal,
}
- return TemplateResponse(request, 'user/user.html', data)
+ return TemplateResponse(request, "user/user.html", data)
+
class Followers(View):
- ''' list of followers view '''
+ """ list of followers view """
+
def get(self, request, username):
- ''' list of followers '''
+ """ list of followers """
try:
user = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -104,21 +109,21 @@ class Followers(View):
return HttpResponseNotFound()
if is_api_request(request):
- return ActivitypubResponse(
- user.to_followers_activity(**request.GET))
+ return ActivitypubResponse(user.to_followers_activity(**request.GET))
data = {
- 'title': '%s: followers' % user.name,
- 'user': user,
- 'is_self': request.user.id == user.id,
- 'followers': user.followers.all(),
+ "user": user,
+ "is_self": request.user.id == user.id,
+ "followers": user.followers.all(),
}
- return TemplateResponse(request, 'user/followers.html', data)
+ return TemplateResponse(request, "user/followers.html", data)
+
class Following(View):
- ''' list of following view '''
+ """ list of following view """
+
def get(self, request, username):
- ''' list of followers '''
+ """ list of followers """
try:
user = get_user_from_username(request.user, username)
except models.User.DoesNotExist:
@@ -129,48 +134,45 @@ class Following(View):
return HttpResponseNotFound()
if is_api_request(request):
- return ActivitypubResponse(
- user.to_following_activity(**request.GET))
+ return ActivitypubResponse(user.to_following_activity(**request.GET))
data = {
- 'title': '%s: following' % user.name,
- 'user': user,
- 'is_self': request.user.id == user.id,
- 'following': user.following.all(),
+ "user": user,
+ "is_self": request.user.id == user.id,
+ "following": user.following.all(),
}
- return TemplateResponse(request, 'user/following.html', data)
+ return TemplateResponse(request, "user/following.html", data)
-@method_decorator(login_required, name='dispatch')
+@method_decorator(login_required, name="dispatch")
class EditUser(View):
- ''' edit user view '''
+ """ edit user view """
+
def get(self, request):
- ''' edit profile page for a user '''
+ """ edit profile page for a user """
data = {
- 'title': 'Edit profile',
- 'form': forms.EditUserForm(instance=request.user),
- 'user': request.user,
+ "form": forms.EditUserForm(instance=request.user),
+ "user": request.user,
}
- return TemplateResponse(request, 'preferences/edit_user.html', data)
+ return TemplateResponse(request, "preferences/edit_user.html", data)
def post(self, request):
- ''' les get fancy with images '''
- form = forms.EditUserForm(
- request.POST, request.FILES, instance=request.user)
+ """ les get fancy with images """
+ form = forms.EditUserForm(request.POST, request.FILES, instance=request.user)
if not form.is_valid():
- data = {'form': form, 'user': request.user}
- return TemplateResponse(request, 'preferences/edit_user.html', data)
+ data = {"form": form, "user": request.user}
+ return TemplateResponse(request, "preferences/edit_user.html", data)
user = form.save(commit=False)
- if 'avatar' in form.files:
+ if "avatar" in form.files:
# crop and resize avatar upload
- image = Image.open(form.files['avatar'])
+ image = Image.open(form.files["avatar"])
image = crop_avatar(image)
# set the name to a hash
- extension = form.files['avatar'].name.split('.')[-1]
- filename = '%s.%s' % (uuid4(), extension)
+ extension = form.files["avatar"].name.split(".")[-1]
+ filename = "%s.%s" % (uuid4(), extension)
user.avatar.save(filename, image)
user.save()
@@ -178,22 +180,27 @@ class EditUser(View):
def crop_avatar(image):
- ''' reduce the size and make an avatar square '''
+ """ reduce the size and make an avatar square """
target_size = 120
width, height = image.size
- thumbnail_scale = height / (width / target_size) if height > width \
+ thumbnail_scale = (
+ height / (width / target_size)
+ if height > width
else width / (height / target_size)
+ )
image.thumbnail([thumbnail_scale, thumbnail_scale])
width, height = image.size
width_diff = width - target_size
height_diff = height - target_size
- cropped = image.crop((
- int(width_diff / 2),
- int(height_diff / 2),
- int(width - (width_diff / 2)),
- int(height - (height_diff / 2))
- ))
+ cropped = image.crop(
+ (
+ int(width_diff / 2),
+ int(height_diff / 2),
+ int(width - (width_diff / 2)),
+ int(height - (height_diff / 2)),
+ )
+ )
output = BytesIO()
cropped.save(output, format=image.format)
return ContentFile(output.getvalue())
diff --git a/bookwyrm/wellknown.py b/bookwyrm/wellknown.py
index 1f6d4ccf..eb014808 100644
--- a/bookwyrm/wellknown.py
+++ b/bookwyrm/wellknown.py
@@ -1,4 +1,4 @@
-''' responds to various requests to /.well-know '''
+""" responds to various requests to /.well-know """
from dateutil.relativedelta import relativedelta
from django.http import HttpResponseNotFound
@@ -10,50 +10,54 @@ from bookwyrm.settings import DOMAIN, VERSION
def webfinger(request):
- ''' allow other servers to ask about a user '''
- if request.method != 'GET':
+ """ allow other servers to ask about a user """
+ if request.method != "GET":
return HttpResponseNotFound()
- resource = request.GET.get('resource')
- if not resource and not resource.startswith('acct:'):
+ resource = request.GET.get("resource")
+ if not resource and not resource.startswith("acct:"):
return HttpResponseNotFound()
- username = resource.replace('acct:', '')
+ username = resource.replace("acct:", "")
try:
user = models.User.objects.get(username=username)
except models.User.DoesNotExist:
- return HttpResponseNotFound('No account found')
+ return HttpResponseNotFound("No account found")
- return JsonResponse({
- 'subject': 'acct:%s' % (user.username),
- 'links': [
- {
- 'rel': 'self',
- 'type': 'application/activity+json',
- 'href': user.remote_id
- }
- ]
- })
+ return JsonResponse(
+ {
+ "subject": "acct:%s" % (user.username),
+ "links": [
+ {
+ "rel": "self",
+ "type": "application/activity+json",
+ "href": user.remote_id,
+ }
+ ],
+ }
+ )
def nodeinfo_pointer(request):
- ''' direct servers to nodeinfo '''
- if request.method != 'GET':
+ """ direct servers to nodeinfo """
+ if request.method != "GET":
return HttpResponseNotFound()
- return JsonResponse({
- 'links': [
- {
- 'rel': 'http://nodeinfo.diaspora.software/ns/schema/2.0',
- 'href': 'https://%s/nodeinfo/2.0' % DOMAIN
- }
- ]
- })
+ return JsonResponse(
+ {
+ "links": [
+ {
+ "rel": "http://nodeinfo.diaspora.software/ns/schema/2.0",
+ "href": "https://%s/nodeinfo/2.0" % DOMAIN,
+ }
+ ]
+ }
+ )
def nodeinfo(request):
- ''' basic info about the server '''
- if request.method != 'GET':
+ """ basic info about the server """
+ if request.method != "GET":
return HttpResponseNotFound()
status_count = models.Status.objects.filter(user__local=True).count()
@@ -61,70 +65,65 @@ def nodeinfo(request):
month_ago = timezone.now() - relativedelta(months=1)
last_month_count = models.User.objects.filter(
- local=True,
- last_active_date__gt=month_ago
+ local=True, last_active_date__gt=month_ago
).count()
six_months_ago = timezone.now() - relativedelta(months=6)
six_month_count = models.User.objects.filter(
- local=True,
- last_active_date__gt=six_months_ago
+ local=True, last_active_date__gt=six_months_ago
).count()
site = models.SiteSettings.get()
- return JsonResponse({
- 'version': '2.0',
- 'software': {
- 'name': 'bookwyrm',
- 'version': VERSION
- },
- 'protocols': [
- 'activitypub'
- ],
- 'usage': {
- 'users': {
- 'total': user_count,
- 'activeMonth': last_month_count,
- 'activeHalfyear': six_month_count,
+ return JsonResponse(
+ {
+ "version": "2.0",
+ "software": {"name": "bookwyrm", "version": VERSION},
+ "protocols": ["activitypub"],
+ "usage": {
+ "users": {
+ "total": user_count,
+ "activeMonth": last_month_count,
+ "activeHalfyear": six_month_count,
+ },
+ "localPosts": status_count,
},
- 'localPosts': status_count,
- },
- 'openRegistrations': site.allow_registration,
- })
+ "openRegistrations": site.allow_registration,
+ }
+ )
def instance_info(request):
- ''' let's talk about your cool unique instance '''
- if request.method != 'GET':
+ """ let's talk about your cool unique instance """
+ if request.method != "GET":
return HttpResponseNotFound()
user_count = models.User.objects.filter(local=True).count()
status_count = models.Status.objects.filter(user__local=True).count()
site = models.SiteSettings.get()
- return JsonResponse({
- 'uri': DOMAIN,
- 'title': site.name,
- 'short_description': '',
- 'description': site.instance_description,
- 'version': '0.0.1',
- 'stats': {
- 'user_count': user_count,
- 'status_count': status_count,
- },
- 'thumbnail': 'https://%s/static/images/logo.png' % DOMAIN,
- 'languages': [
- 'en'
- ],
- 'registrations': site.allow_registration,
- 'approval_required': False,
- })
+ return JsonResponse(
+ {
+ "uri": DOMAIN,
+ "title": site.name,
+ "short_description": "",
+ "description": site.instance_description,
+ "version": "0.0.1",
+ "stats": {
+ "user_count": user_count,
+ "status_count": status_count,
+ },
+ "thumbnail": "https://%s/static/images/logo.png" % DOMAIN,
+ "languages": ["en"],
+ "registrations": site.allow_registration,
+ "approval_required": False,
+ }
+ )
def peers(request):
- ''' list of federated servers this instance connects with '''
- if request.method != 'GET':
+ """ list of federated servers this instance connects with """
+ if request.method != "GET":
return HttpResponseNotFound()
- names = models.FederatedServer.objects.values_list('server_name', flat=True)
+ names = models.FederatedServer.objects.values_list("server_name", flat=True)
return JsonResponse(list(names), safe=False)
diff --git a/bw-dev b/bw-dev
index 27dbc4c3..74c42fbb 100755
--- a/bw-dev
+++ b/bw-dev
@@ -35,6 +35,10 @@ function initdb {
execweb python manage.py initdb
}
+function makeitblack {
+ runweb black celerywyrm bookwyrm
+}
+
CMD=$1
shift
@@ -91,10 +95,10 @@ case "$CMD" in
execweb python manage.py collectstatic --no-input
;;
makemessages)
- django-admin makemessages --extension html --ignore=venv3 $@
+ execweb django-admin makemessages --no-wrap --ignore=venv3 $@
;;
compilemessages)
- django-admin compilemessages --ignore venv3 $@
+ execweb django-admin compilemessages --ignore venv3 $@
;;
build)
docker-compose build
@@ -102,7 +106,10 @@ case "$CMD" in
clean)
clean
;;
+ black)
+ makeitblack
+ ;;
*)
- echo "Unrecognised command. Try: build, clean, up, initdb, resetdb, makemigrations, migrate, bash, shell, dbshell, restart_celery, test, pytest, test_report"
+ echo "Unrecognised command. Try: build, clean, up, initdb, resetdb, makemigrations, migrate, bash, shell, dbshell, restart_celery, test, pytest, test_report, black"
;;
esac
diff --git a/celerywyrm/__init__.py b/celerywyrm/__init__.py
index 1c8fa15d..fe0c87ff 100644
--- a/celerywyrm/__init__.py
+++ b/celerywyrm/__init__.py
@@ -1,8 +1,8 @@
-''' we need this file to initialize celery '''
+""" we need this file to initialize celery """
from __future__ import absolute_import, unicode_literals
# This will make sure the app is always imported when
# Django starts so that shared_task will use this app.
from .celery import app as celery_app
-__all__ = ('celery_app',)
+__all__ = ("celery_app",)
diff --git a/celerywyrm/asgi.py b/celerywyrm/asgi.py
index c03a6ec6..0f0b0021 100644
--- a/celerywyrm/asgi.py
+++ b/celerywyrm/asgi.py
@@ -11,6 +11,6 @@ import os
from django.core.asgi import get_asgi_application
-os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'celerywyrm.settings')
+os.environ.setdefault("DJANGO_SETTINGS_MODULE", "celerywyrm.settings")
application = get_asgi_application()
diff --git a/celerywyrm/celery.py b/celerywyrm/celery.py
index 2937ef0f..4af8e281 100644
--- a/celerywyrm/celery.py
+++ b/celerywyrm/celery.py
@@ -1,4 +1,4 @@
-''' configures celery for task management '''
+""" configures celery for task management """
from __future__ import absolute_import, unicode_literals
import os
@@ -7,23 +7,22 @@ from . import settings
# set the default Django settings module for the 'celery' program.
-os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'celerywyrm.settings')
+os.environ.setdefault("DJANGO_SETTINGS_MODULE", "celerywyrm.settings")
-app = Celery('celerywyrm')
+app = Celery("celerywyrm")
# Using a string here means the worker doesn't have to serialize
# the configuration object to child processes.
# - namespace='CELERY' means all celery-related configuration keys
# should have a `CELERY_` prefix.
-app.config_from_object('django.conf:settings', namespace='CELERY')
+app.config_from_object("django.conf:settings", namespace="CELERY")
# Load task modules from all registered Django app configs.
app.autodiscover_tasks()
-app.autodiscover_tasks(['bookwyrm'], related_name='activitypub.base_activity')
-app.autodiscover_tasks(['bookwyrm'], related_name='broadcast')
-app.autodiscover_tasks(
- ['bookwyrm'], related_name='connectors.abstract_connector')
-app.autodiscover_tasks(['bookwyrm'], related_name='emailing')
-app.autodiscover_tasks(['bookwyrm'], related_name='goodreads_import')
-app.autodiscover_tasks(['bookwyrm'], related_name='models.user')
-app.autodiscover_tasks(['bookwyrm'], related_name='views.inbox')
+app.autodiscover_tasks(["bookwyrm"], related_name="activitypub.base_activity")
+app.autodiscover_tasks(["bookwyrm"], related_name="broadcast")
+app.autodiscover_tasks(["bookwyrm"], related_name="connectors.abstract_connector")
+app.autodiscover_tasks(["bookwyrm"], related_name="emailing")
+app.autodiscover_tasks(["bookwyrm"], related_name="goodreads_import")
+app.autodiscover_tasks(["bookwyrm"], related_name="models.user")
+app.autodiscover_tasks(["bookwyrm"], related_name="views.inbox")
diff --git a/celerywyrm/settings.py b/celerywyrm/settings.py
index 92986d8e..7591163b 100644
--- a/celerywyrm/settings.py
+++ b/celerywyrm/settings.py
@@ -16,31 +16,31 @@ from environs import Env
env = Env()
# emailing
-EMAIL_HOST = env('EMAIL_HOST')
-EMAIL_PORT = env('EMAIL_PORT')
-EMAIL_HOST_USER = env('EMAIL_HOST_USER')
-EMAIL_HOST_PASSWORD = env('EMAIL_HOST_PASSWORD')
-EMAIL_USE_TLS = env('EMAIL_USE_TLS')
+EMAIL_HOST = env("EMAIL_HOST")
+EMAIL_PORT = env("EMAIL_PORT")
+EMAIL_HOST_USER = env("EMAIL_HOST_USER")
+EMAIL_HOST_PASSWORD = env("EMAIL_HOST_PASSWORD")
+EMAIL_USE_TLS = env("EMAIL_USE_TLS")
# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
# celery/rebbitmq
-CELERY_BROKER_URL = env('CELERY_BROKER')
-CELERY_ACCEPT_CONTENT = ['json']
-CELERY_TASK_SERIALIZER = 'json'
-CELERY_RESULT_BACKEND = 'redis'
+CELERY_BROKER_URL = env("CELERY_BROKER")
+CELERY_ACCEPT_CONTENT = ["json"]
+CELERY_TASK_SERIALIZER = "json"
+CELERY_RESULT_BACKEND = "redis"
# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/3.0/howto/deployment/checklist/
# SECURITY WARNING: keep the secret key used in production secret!
-SECRET_KEY = '0a^0gpwjc1ap+lb$dinin=efc@e&_0%102$o3(>9e7lndiaw'
+SECRET_KEY = "0a^0gpwjc1ap+lb$dinin=efc@e&_0%102$o3(>9e7lndiaw"
# SECURITY WARNING: don't run with debug turned on in production!
-DEBUG = env.bool('DEBUG', True)
+DEBUG = env.bool("DEBUG", True)
ALLOWED_HOSTS = []
@@ -49,71 +49,69 @@ ALLOWED_HOSTS = []
# Application definition
INSTALLED_APPS = [
- 'django.contrib.admin',
- 'django.contrib.auth',
- 'django.contrib.contenttypes',
- 'django.contrib.sessions',
- 'django.contrib.messages',
- 'django.contrib.staticfiles',
- 'celerywyrm',
- 'bookwyrm',
- 'celery',
+ "django.contrib.admin",
+ "django.contrib.auth",
+ "django.contrib.contenttypes",
+ "django.contrib.sessions",
+ "django.contrib.messages",
+ "django.contrib.staticfiles",
+ "celerywyrm",
+ "bookwyrm",
+ "celery",
]
MIDDLEWARE = [
- 'django.middleware.security.SecurityMiddleware',
- 'django.contrib.sessions.middleware.SessionMiddleware',
- 'django.middleware.common.CommonMiddleware',
- 'django.middleware.csrf.CsrfViewMiddleware',
- 'django.contrib.auth.middleware.AuthenticationMiddleware',
- 'django.contrib.messages.middleware.MessageMiddleware',
- 'django.middleware.clickjacking.XFrameOptionsMiddleware',
+ "django.middleware.security.SecurityMiddleware",
+ "django.contrib.sessions.middleware.SessionMiddleware",
+ "django.middleware.common.CommonMiddleware",
+ "django.middleware.csrf.CsrfViewMiddleware",
+ "django.contrib.auth.middleware.AuthenticationMiddleware",
+ "django.contrib.messages.middleware.MessageMiddleware",
+ "django.middleware.clickjacking.XFrameOptionsMiddleware",
]
-ROOT_URLCONF = 'celerywyrm.urls'
+ROOT_URLCONF = "celerywyrm.urls"
TEMPLATES = [
{
- 'BACKEND': 'django.template.backends.django.DjangoTemplates',
- 'DIRS': [],
- 'APP_DIRS': True,
- 'OPTIONS': {
- 'context_processors': [
- 'django.template.context_processors.debug',
- 'django.template.context_processors.request',
- 'django.contrib.auth.context_processors.auth',
- 'django.contrib.messages.context_processors.messages',
+ "BACKEND": "django.template.backends.django.DjangoTemplates",
+ "DIRS": [],
+ "APP_DIRS": True,
+ "OPTIONS": {
+ "context_processors": [
+ "django.template.context_processors.debug",
+ "django.template.context_processors.request",
+ "django.contrib.auth.context_processors.auth",
+ "django.contrib.messages.context_processors.messages",
],
},
},
]
-WSGI_APPLICATION = 'celerywyrm.wsgi.application'
+WSGI_APPLICATION = "celerywyrm.wsgi.application"
# Database
# https://docs.djangoproject.com/en/3.0/ref/settings/#databases
-BOOKWYRM_DATABASE_BACKEND = env('BOOKWYRM_DATABASE_BACKEND', 'postgres')
+BOOKWYRM_DATABASE_BACKEND = env("BOOKWYRM_DATABASE_BACKEND", "postgres")
BOOKWYRM_DBS = {
- 'postgres': {
- 'ENGINE': 'django.db.backends.postgresql_psycopg2',
- 'NAME': env('POSTGRES_DB', 'fedireads'),
- 'USER': env('POSTGRES_USER', 'fedireads'),
- 'PASSWORD': env('POSTGRES_PASSWORD', 'fedireads'),
- 'HOST': env('POSTGRES_HOST', ''),
- 'PORT': 5432
+ "postgres": {
+ "ENGINE": "django.db.backends.postgresql_psycopg2",
+ "NAME": env("POSTGRES_DB", "fedireads"),
+ "USER": env("POSTGRES_USER", "fedireads"),
+ "PASSWORD": env("POSTGRES_PASSWORD", "fedireads"),
+ "HOST": env("POSTGRES_HOST", ""),
+ "PORT": 5432,
+ },
+ "sqlite": {
+ "ENGINE": "django.db.backends.sqlite3",
+ "NAME": os.path.join(BASE_DIR, "fedireads.db"),
},
- 'sqlite': {
- 'ENGINE': 'django.db.backends.sqlite3',
- 'NAME': os.path.join(BASE_DIR, 'fedireads.db')
- }
}
-DATABASES = {
- 'default': BOOKWYRM_DBS[BOOKWYRM_DATABASE_BACKEND]
-}
+DATABASES = {"default": BOOKWYRM_DBS[BOOKWYRM_DATABASE_BACKEND]}
# Password validation
@@ -121,16 +119,16 @@ DATABASES = {
AUTH_PASSWORD_VALIDATORS = [
{
- 'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',
+ "NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",
},
{
- 'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',
+ "NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",
},
{
- 'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',
+ "NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",
},
{
- 'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',
+ "NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",
},
]
@@ -138,9 +136,9 @@ AUTH_PASSWORD_VALIDATORS = [
# Internationalization
# https://docs.djangoproject.com/en/3.0/topics/i18n/
-LANGUAGE_CODE = 'en-us'
+LANGUAGE_CODE = "en-us"
-TIME_ZONE = 'UTC'
+TIME_ZONE = "UTC"
USE_I18N = False
@@ -152,7 +150,7 @@ USE_TZ = True
# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/3.0/howto/static-files/
-STATIC_URL = '/static/'
-STATIC_ROOT = os.path.join(BASE_DIR, env('STATIC_ROOT', 'static'))
-MEDIA_URL = '/images/'
-MEDIA_ROOT = os.path.join(BASE_DIR, env('MEDIA_ROOT', 'images'))
+STATIC_URL = "/static/"
+STATIC_ROOT = os.path.join(BASE_DIR, env("STATIC_ROOT", "static"))
+MEDIA_URL = "/images/"
+MEDIA_ROOT = os.path.join(BASE_DIR, env("MEDIA_ROOT", "images"))
diff --git a/celerywyrm/urls.py b/celerywyrm/urls.py
index 21c8dee8..394c0ef0 100644
--- a/celerywyrm/urls.py
+++ b/celerywyrm/urls.py
@@ -20,5 +20,5 @@ from django.urls import path
from celerywyrm import settings
urlpatterns = [
- path('admin/', admin.site.urls),
+ path("admin/", admin.site.urls),
] + static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)
diff --git a/celerywyrm/wsgi.py b/celerywyrm/wsgi.py
index 6de8d633..7ccf36a9 100644
--- a/celerywyrm/wsgi.py
+++ b/celerywyrm/wsgi.py
@@ -11,6 +11,6 @@ import os
from django.core.wsgi import get_wsgi_application
-os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'celerywyrm.settings')
+os.environ.setdefault("DJANGO_SETTINGS_MODULE", "celerywyrm.settings")
application = get_wsgi_application()
diff --git a/locale/de_DE/LC_MESSAGES/django.mo b/locale/de_DE/LC_MESSAGES/django.mo
new file mode 100644
index 00000000..16e093c6
Binary files /dev/null and b/locale/de_DE/LC_MESSAGES/django.mo differ
diff --git a/locale/de_DE/LC_MESSAGES/django.po b/locale/de_DE/LC_MESSAGES/django.po
new file mode 100644
index 00000000..a335f2ce
--- /dev/null
+++ b/locale/de_DE/LC_MESSAGES/django.po
@@ -0,0 +1,1884 @@
+# German language text for the bookwyrm UI
+# Copyright (C) 2021 Mouse Reeve
+# This file is distributed under the same license as the BookWyrm package.
+# Mouse Reeve , 2021
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: 0.0.1\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2021-03-07 23:40+0000\n"
+"PO-Revision-Date: 2021-03-02 17:19-0800\n"
+"Last-Translator: Mouse Reeve \n"
+"Language-Team: English \n"
+"Language: de_DE\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=(n != 1);\n"
+
+#: bookwyrm/forms.py:185
+msgid "One Day"
+msgstr "Ein Tag"
+
+#: bookwyrm/forms.py:186
+msgid "One Week"
+msgstr "Eine Woche"
+
+#: bookwyrm/forms.py:187
+msgid "One Month"
+msgstr "Ein Monat"
+
+#: bookwyrm/forms.py:188
+msgid "Does Not Expire"
+msgstr "Läuft nicht aus"
+
+#: bookwyrm/forms.py:190
+#, python-format
+msgid "%(count)d uses"
+msgstr "%(count)d Benutzungen"
+
+#: bookwyrm/forms.py:192
+#, fuzzy
+#| msgid "Unlisted"
+msgid "Unlimited"
+msgstr "Ungelistet"
+
+#: bookwyrm/models/fields.py:24
+#, python-format
+msgid "%(value)s is not a valid remote_id"
+msgstr "%(value)s ist keine gültige remote_id"
+
+#: bookwyrm/models/fields.py:33 bookwyrm/models/fields.py:42
+#, python-format
+msgid "%(value)s is not a valid username"
+msgstr "%(value)s ist kein gültiger Username"
+
+#: bookwyrm/models/fields.py:164 bookwyrm/templates/layout.html:152
+msgid "username"
+msgstr "Username"
+
+#: bookwyrm/models/fields.py:169
+msgid "A user with that username already exists."
+msgstr "Dieser Benutzename ist bereits vergeben."
+
+#: bookwyrm/settings.py:142
+msgid "English"
+msgstr "Englisch"
+
+#: bookwyrm/settings.py:143
+msgid "German"
+msgstr "Deutsch"
+
+#: bookwyrm/settings.py:144
+msgid "Spanish"
+msgstr ""
+
+#: bookwyrm/settings.py:145
+msgid "French"
+msgstr "Französisch"
+
+#: bookwyrm/settings.py:146
+msgid "Simplified Chinese"
+msgstr "Vereinfachtes Chinesisch"
+
+#: bookwyrm/templates/author.html:16 bookwyrm/templates/author.html:17
+#: bookwyrm/templates/edit_author.html:5
+msgid "Edit Author"
+msgstr "Autor*in editieren"
+
+#: bookwyrm/templates/author.html:32
+msgid "Wikipedia"
+msgstr ""
+
+#: bookwyrm/templates/author.html:37
+#, python-format
+msgid "Books by %(name)s"
+msgstr "Bücher von %(name)s"
+
+#: bookwyrm/templates/book.html:21
+#: bookwyrm/templates/discover/large-book.html:12
+#: bookwyrm/templates/discover/small-book.html:9
+msgid "by"
+msgstr "von"
+
+#: bookwyrm/templates/book.html:29 bookwyrm/templates/book.html:30
+#: bookwyrm/templates/edit_book.html:5
+msgid "Edit Book"
+msgstr "Buch editieren"
+
+#: bookwyrm/templates/book.html:45
+msgid "Add cover"
+msgstr "Cover hinzufügen"
+
+#: bookwyrm/templates/book.html:51 bookwyrm/templates/lists/list.html:89
+msgid "Add"
+msgstr "Hinzufügen"
+
+#: bookwyrm/templates/book.html:60
+msgid "ISBN:"
+msgstr ""
+
+#: bookwyrm/templates/book.html:67 bookwyrm/templates/edit_book.html:107
+msgid "OCLC Number:"
+msgstr "OCLC Nummer:"
+
+#: bookwyrm/templates/book.html:74 bookwyrm/templates/edit_book.html:111
+msgid "ASIN:"
+msgstr ""
+
+#: bookwyrm/templates/book.html:84
+#, fuzzy, python-format
+#| msgid "%(format)s, %(pages)s pages"
+msgid "%(format)s, %(pages)s pages"
+msgstr "%(format)s, %(book.pages)s Seiten"
+
+#: bookwyrm/templates/book.html:86
+#, fuzzy, python-format
+#| msgid "%(pages)s pages"
+msgid "%(pages)s pages"
+msgstr "%(book.pages)s Seiten"
+
+#: bookwyrm/templates/book.html:91
+msgid "View on OpenLibrary"
+msgstr "In OpenLibrary ansehen"
+
+#: bookwyrm/templates/book.html:100
+#, python-format
+msgid "(%(review_count)s review)"
+msgid_plural "(%(review_count)s reviews)"
+msgstr[0] "(%(review_count)s Bewertung)"
+msgstr[1] "(%(review_count)s Bewertungen)"
+
+#: bookwyrm/templates/book.html:106
+msgid "Add Description"
+msgstr "Beschreibung hinzufügen"
+
+#: bookwyrm/templates/book.html:113 bookwyrm/templates/edit_book.html:39
+#: bookwyrm/templates/lists/form.html:12
+msgid "Description:"
+msgstr "Beschreibung:"
+
+#: bookwyrm/templates/book.html:117 bookwyrm/templates/edit_author.html:78
+#: bookwyrm/templates/edit_book.html:120 bookwyrm/templates/lists/form.html:42
+#: bookwyrm/templates/preferences/edit_user.html:50
+#: bookwyrm/templates/settings/site.html:89
+#: bookwyrm/templates/snippets/progress_update.html:21
+#: bookwyrm/templates/snippets/readthrough.html:64
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:42
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:34
+msgid "Save"
+msgstr "Speichern"
+
+#: bookwyrm/templates/book.html:118 bookwyrm/templates/book.html:167
+#: bookwyrm/templates/edit_author.html:79 bookwyrm/templates/edit_book.html:121
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:17
+#: bookwyrm/templates/snippets/goal_form.html:32
+#: bookwyrm/templates/snippets/readthrough.html:65
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:43
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:35
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:28
+msgid "Cancel"
+msgstr "Abbrechen"
+
+#: bookwyrm/templates/book.html:127
+#, fuzzy, python-format
+#| msgid "%(title)s by "
+msgid "%(count)s editions "
+msgstr "%(title)s von"
+
+#: bookwyrm/templates/book.html:135
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "This edition is on your %(shelf_name)s shelf."
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/book.html:141
+#, fuzzy, python-format
+#| msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgid "A different edition of this book is on your %(shelf_name)s shelf."
+msgstr "hat %(book_title)s zu deiner Liste \"%(list_name)s \" Hinzugefügt"
+
+#: bookwyrm/templates/book.html:150
+msgid "Your reading activity"
+msgstr "Deine Leseaktivität"
+
+#: bookwyrm/templates/book.html:152
+msgid "Add read dates"
+msgstr "Lesedaten hinzufügen"
+
+#: bookwyrm/templates/book.html:157
+msgid "You don't have any reading activity for this book."
+msgstr "Du hast keine Leseaktivität für dieses Buch."
+
+#: bookwyrm/templates/book.html:164
+msgid "Create"
+msgstr "Erstellen"
+
+#: bookwyrm/templates/book.html:186
+msgid "Tags"
+msgstr ""
+
+#: bookwyrm/templates/book.html:190 bookwyrm/templates/snippets/tag.html:18
+msgid "Add tag"
+msgstr "Tag hinzufügen"
+
+#: bookwyrm/templates/book.html:207
+msgid "Subjects"
+msgstr "Themen"
+
+#: bookwyrm/templates/book.html:218
+msgid "Places"
+msgstr "Orte"
+
+#: bookwyrm/templates/book.html:229 bookwyrm/templates/layout.html:64
+#: bookwyrm/templates/lists/lists.html:4 bookwyrm/templates/lists/lists.html:9
+#: bookwyrm/templates/search_results.html:92
+#: bookwyrm/templates/user/user_layout.html:62
+msgid "Lists"
+msgstr "Listen"
+
+#: bookwyrm/templates/book.html:258
+msgid "rated it"
+msgstr "bewertet"
+
+#: bookwyrm/templates/components/inline_form.html:8
+#: bookwyrm/templates/feed/feed_layout.html:54
+msgid "Close"
+msgstr "Schließen"
+
+#: bookwyrm/templates/discover/about.html:7
+#, python-format
+msgid "About %(site_name)s"
+msgstr "Über %(site_name)s"
+
+#: bookwyrm/templates/discover/about.html:10
+#: bookwyrm/templates/discover/about.html:20
+msgid "Code of Conduct"
+msgstr ""
+
+#: bookwyrm/templates/discover/about.html:13
+#: bookwyrm/templates/discover/about.html:29
+msgid "Privacy Policy"
+msgstr "Datenschutzerklärung"
+
+#: bookwyrm/templates/discover/discover.html:6
+msgid "Recent Books"
+msgstr "Aktive Bücher"
+
+#: bookwyrm/templates/discover/landing_layout.html:5
+msgid "Welcome"
+msgstr "Willkommen"
+
+#: bookwyrm/templates/discover/landing_layout.html:17
+msgid "Decentralized"
+msgstr "Dezentral"
+
+#: bookwyrm/templates/discover/landing_layout.html:23
+msgid "Friendly"
+msgstr "Freundlich"
+
+#: bookwyrm/templates/discover/landing_layout.html:29
+msgid "Anti-Corporate"
+msgstr ""
+
+#: bookwyrm/templates/discover/landing_layout.html:44
+#, python-format
+msgid "Join %(name)s"
+msgstr "Tritt %(name)s bei"
+
+#: bookwyrm/templates/discover/landing_layout.html:49
+#: bookwyrm/templates/login.html:48
+msgid "This instance is closed"
+msgstr "Diese Instanz ist geschlossen"
+
+#: bookwyrm/templates/discover/landing_layout.html:55
+msgid "Your Account"
+msgstr "Dein Account"
+
+#: bookwyrm/templates/edit_author.html:13 bookwyrm/templates/edit_book.html:13
+msgid "Added:"
+msgstr "Hinzugefügt:"
+
+#: bookwyrm/templates/edit_author.html:14 bookwyrm/templates/edit_book.html:14
+msgid "Updated:"
+msgstr "Aktualisiert:"
+
+#: bookwyrm/templates/edit_author.html:15 bookwyrm/templates/edit_book.html:15
+msgid "Last edited by:"
+msgstr "Zuletzt bearbeitet von:"
+
+#: bookwyrm/templates/edit_author.html:31 bookwyrm/templates/edit_book.html:30
+msgid "Metadata"
+msgstr "Metadaten"
+
+#: bookwyrm/templates/edit_author.html:32 bookwyrm/templates/lists/form.html:8
+#: bookwyrm/templates/user/create_shelf_form.html:13
+#: bookwyrm/templates/user/edit_shelf_form.html:14
+msgid "Name:"
+msgstr ""
+
+#: bookwyrm/templates/edit_author.html:37
+msgid "Bio:"
+msgstr ""
+
+#: bookwyrm/templates/edit_author.html:42
+msgid "Wikipedia link:"
+msgstr "Wikipedialink:"
+
+#: bookwyrm/templates/edit_author.html:47
+msgid "Birth date:"
+msgstr "Geburtsdatum:"
+
+#: bookwyrm/templates/edit_author.html:52
+msgid "Death date:"
+msgstr "Todesdatum:"
+
+#: bookwyrm/templates/edit_author.html:58
+msgid "Author Identifiers"
+msgstr "Autor*innenidentifikatoren"
+
+#: bookwyrm/templates/edit_author.html:59 bookwyrm/templates/edit_book.html:103
+msgid "Openlibrary key:"
+msgstr ""
+
+#: bookwyrm/templates/edit_author.html:64
+msgid "Librarything key:"
+msgstr ""
+
+#: bookwyrm/templates/edit_author.html:69
+msgid "Goodreads key:"
+msgstr ""
+
+#: bookwyrm/templates/edit_book.html:31
+msgid "Title:"
+msgstr "Titel:"
+
+#: bookwyrm/templates/edit_book.html:35
+msgid "Subtitle:"
+msgstr "Untertitel:"
+
+#: bookwyrm/templates/edit_book.html:43
+msgid "Series:"
+msgstr "Serie:"
+
+#: bookwyrm/templates/edit_book.html:47
+msgid "Series number:"
+msgstr "Seriennummer:"
+
+#: bookwyrm/templates/edit_book.html:51
+msgid "First published date:"
+msgstr "Erstveröffentlichungsdatum:"
+
+#: bookwyrm/templates/edit_book.html:55
+msgid "Published date:"
+msgstr "Veröffentlichungsdatum:"
+
+#: bookwyrm/templates/edit_book.html:68
+#: bookwyrm/templates/snippets/shelf.html:9
+msgid "Cover"
+msgstr ""
+
+#: bookwyrm/templates/edit_book.html:78
+msgid "Physical Properties"
+msgstr "Physikalische Eigenschaften"
+
+#: bookwyrm/templates/edit_book.html:79
+msgid "Format:"
+msgstr ""
+
+#: bookwyrm/templates/edit_book.html:87
+msgid "Pages:"
+msgstr "Seiten:"
+
+#: bookwyrm/templates/edit_book.html:94
+msgid "Book Identifiers"
+msgstr "Buchidentifikatoren"
+
+#: bookwyrm/templates/edit_book.html:95
+msgid "ISBN 13:"
+msgstr ""
+
+#: bookwyrm/templates/edit_book.html:99
+msgid "ISBN 10:"
+msgstr ""
+
+#: bookwyrm/templates/editions.html:5
+#, python-format
+msgid "Editions of %(book_title)s"
+msgstr "Editionen von %(book_title)s"
+
+#: bookwyrm/templates/editions.html:9
+#, python-format
+msgid "Editions of \"%(work_title)s\" "
+msgstr "Editionen von \"%(work_title)s\" "
+
+#: bookwyrm/templates/error.html:4
+msgid "Oops!"
+msgstr "Ups!"
+
+#: bookwyrm/templates/error.html:8
+msgid "Server Error"
+msgstr ""
+
+#: bookwyrm/templates/error.html:9
+msgid "Something went wrong! Sorry about that."
+msgstr "Etwas lief schief. Entschuldigung!"
+
+#: bookwyrm/templates/feed/direct_messages.html:8
+#, python-format
+msgid "Direct Messages with %(username)s "
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/feed/direct_messages.html:10
+#: bookwyrm/templates/layout.html:87
+msgid "Direct Messages"
+msgstr "Direktnachrichten"
+
+#: bookwyrm/templates/feed/direct_messages.html:13
+msgid "All messages"
+msgstr "Alle Nachrichten"
+
+#: bookwyrm/templates/feed/direct_messages.html:22
+msgid "You have no messages right now."
+msgstr "Du hast momentan keine Nachrichten."
+
+#: bookwyrm/templates/feed/feed.html:6
+#, python-format
+msgid "%(tab_title)s Timeline"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed.html:10 bookwyrm/views/feed.py:33
+msgid "Home"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed.html:13 bookwyrm/views/feed.py:37
+msgid "Local"
+msgstr "Lokal"
+
+#: bookwyrm/templates/feed/feed.html:16 bookwyrm/views/feed.py:41
+msgid "Federated"
+msgstr "Föderiert"
+
+#: bookwyrm/templates/feed/feed.html:24
+msgid "Announcements"
+msgstr "Ankündigungen"
+
+#: bookwyrm/templates/feed/feed.html:32
+msgid "There aren't any activities right now! Try following a user to get started"
+msgstr "Hier sind noch keine Aktivitäten! Folge anderen, um loszulegen"
+
+#: bookwyrm/templates/feed/feed_layout.html:5
+msgid "Updates"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed_layout.html:11
+msgid "Your books"
+msgstr "Deine Bücher"
+
+#: bookwyrm/templates/feed/feed_layout.html:13
+msgid "There are no books here right now! Try searching for a book to get started"
+msgstr "Hier sind noch keine Bücher! Versuche nach Büchern zu suchen um loszulegen"
+
+#: bookwyrm/templates/feed/feed_layout.html:23
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Read"
+msgid "To Read"
+msgstr "Auf der Leseliste"
+
+#: bookwyrm/templates/feed/feed_layout.html:24
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Start reading"
+msgid "Currently Reading"
+msgstr "Gerade lesend"
+
+#: bookwyrm/templates/feed/feed_layout.html:25
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:11
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Read"
+msgstr "Gelesen"
+
+#: bookwyrm/templates/feed/feed_layout.html:76 bookwyrm/templates/goal.html:26
+#: bookwyrm/templates/snippets/goal_card.html:6
+#, python-format
+msgid "%(year)s Reading Goal"
+msgstr "%(year)s Leseziel"
+
+#: bookwyrm/templates/feed/status.html:8
+msgid "Back"
+msgstr "Zurück"
+
+#: bookwyrm/templates/goal.html:7
+#, python-format
+msgid "%(year)s Reading Progress"
+msgstr "%(year)s Lesefortschritt"
+
+#: bookwyrm/templates/goal.html:11
+msgid "Edit Goal"
+msgstr "Ziel bearbeiten"
+
+#: bookwyrm/templates/goal.html:30
+#: bookwyrm/templates/snippets/goal_card.html:13
+#, python-format
+msgid "Set a goal for how many books you'll finish reading in %(year)s, and track your progress throughout the year."
+msgstr "Setze dir ein Ziel, wie viele Bücher du %(year)s lesen wirst und behalte deinen Fortschritt über's Jahr im Auge."
+
+#: bookwyrm/templates/goal.html:39
+#, python-format
+msgid "%(name)s hasn't set a reading goal for %(year)s."
+msgstr "%(name)s hat sich für %(year)s kein Leseziel gesetzt."
+
+#: bookwyrm/templates/goal.html:51
+#, python-format
+msgid "Your %(year)s Books"
+msgstr "Deine Bücher %(year)s"
+
+#: bookwyrm/templates/goal.html:53
+#, python-format
+msgid "%(username)s's %(year)s Books"
+msgstr "%(username)ss %(year)s Bücher"
+
+#: bookwyrm/templates/import.html:5 bookwyrm/templates/import.html:9
+#: bookwyrm/templates/layout.html:102
+msgid "Import Books"
+msgstr "Bücher importieren"
+
+#: bookwyrm/templates/import.html:14
+msgid "Data source"
+msgstr "Datenquelle"
+
+#: bookwyrm/templates/import.html:32
+msgid "Include reviews"
+msgstr "Bewertungen importieren"
+
+#: bookwyrm/templates/import.html:37
+msgid "Privacy setting for imported reviews:"
+msgstr "Datenschutzeinstellung für importierte Bewertungen"
+
+#: bookwyrm/templates/import.html:41
+msgid "Import"
+msgstr "Importieren"
+
+#: bookwyrm/templates/import.html:46
+msgid "Recent Imports"
+msgstr "Aktuelle Importe"
+
+#: bookwyrm/templates/import.html:48
+msgid "No recent imports"
+msgstr "Keine aktuellen Importe"
+
+#: bookwyrm/templates/import_status.html:6
+#: bookwyrm/templates/import_status.html:10
+msgid "Import Status"
+msgstr "Importstatus"
+
+#: bookwyrm/templates/import_status.html:13
+msgid "Import started:"
+msgstr "Import gestartet:"
+
+#: bookwyrm/templates/import_status.html:17
+msgid "Import completed:"
+msgstr "Import abgeschlossen:"
+
+#: bookwyrm/templates/import_status.html:20
+msgid "TASK FAILED"
+msgstr "AUFGABE GESCHEITERT"
+
+#: bookwyrm/templates/import_status.html:26
+msgid "Import still in progress."
+msgstr "Import läuft noch."
+
+#: bookwyrm/templates/import_status.html:28
+msgid "(Hit reload to update!)"
+msgstr "(Aktualisiere für ein Update!)"
+
+#: bookwyrm/templates/import_status.html:35
+msgid "Failed to load"
+msgstr "Laden fehlgeschlagen"
+
+#: bookwyrm/templates/import_status.html:44
+#, python-format
+msgid "Jump to the bottom of the list to select the %(failed_count)s items which failed to import."
+msgstr ""
+
+#: bookwyrm/templates/import_status.html:79
+msgid "Select all"
+msgstr "Alle auswählen"
+
+#: bookwyrm/templates/import_status.html:82
+msgid "Retry items"
+msgstr "Punkte erneut versuchen"
+
+#: bookwyrm/templates/import_status.html:108
+msgid "Successfully imported"
+msgstr "Erfolgreich importiert"
+
+#: bookwyrm/templates/import_status.html:112
+#: bookwyrm/templates/lists/curate.html:14
+msgid "Book"
+msgstr "Buch"
+
+#: bookwyrm/templates/import_status.html:115
+#: bookwyrm/templates/snippets/create_status_form.html:10
+#: bookwyrm/templates/snippets/shelf.html:10
+msgid "Title"
+msgstr "Titel"
+
+#: bookwyrm/templates/import_status.html:118
+#: bookwyrm/templates/snippets/shelf.html:11
+msgid "Author"
+msgstr "Autor*in"
+
+#: bookwyrm/templates/import_status.html:141
+msgid "Imported"
+msgstr "Importiert"
+
+#: bookwyrm/templates/invite.html:4 bookwyrm/templates/invite.html:12
+#: bookwyrm/templates/login.html:43
+msgid "Create an Account"
+msgstr "Erstelle einen Account"
+
+#: bookwyrm/templates/invite.html:21
+msgid "Permission Denied"
+msgstr "Zugiff verweigert"
+
+#: bookwyrm/templates/invite.html:22
+msgid "Sorry! This invite code is no longer valid."
+msgstr "Sorry! Dieser Einladecode ist mehr gültig."
+
+#: bookwyrm/templates/isbn_search_results.html:4
+#: bookwyrm/templates/search_results.html:4
+msgid "Search Results"
+msgstr "Suchergebnisse"
+
+#: bookwyrm/templates/isbn_search_results.html:9
+#: bookwyrm/templates/search_results.html:9
+#, python-format
+msgid "Search Results for \"%(query)s\""
+msgstr "Suchergebnisse für \"%(query)s\""
+
+#: bookwyrm/templates/isbn_search_results.html:14
+#: bookwyrm/templates/search_results.html:14
+msgid "Matching Books"
+msgstr "Passende Bücher"
+
+#: bookwyrm/templates/isbn_search_results.html:17
+#: bookwyrm/templates/search_results.html:17
+#, python-format
+msgid "No books found for \"%(query)s\""
+msgstr "Keine Bücher für \"%(query)s\" gefunden"
+
+#: bookwyrm/templates/layout.html:33
+msgid "Search for a book or user"
+msgstr "Suche nach Buch oder Benutzer*in"
+
+#: bookwyrm/templates/layout.html:37 bookwyrm/templates/layout.html:38
+#: bookwyrm/templates/lists/list.html:62
+msgid "Search"
+msgstr "Suche"
+
+#: bookwyrm/templates/layout.html:47 bookwyrm/templates/layout.html:48
+msgid "Main navigation menu"
+msgstr "Navigationshauptmenü"
+
+#: bookwyrm/templates/layout.html:58
+msgid "Your shelves"
+msgstr "Deine Regale"
+
+#: bookwyrm/templates/layout.html:61
+msgid "Feed"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:92
+#: bookwyrm/templates/preferences/preferences_layout.html:14
+msgid "Profile"
+msgstr "Profil"
+
+#: bookwyrm/templates/layout.html:97
+msgid "Settings"
+msgstr "Einstellungen"
+
+#: bookwyrm/templates/layout.html:111
+#: bookwyrm/templates/settings/admin_layout.html:19
+#: bookwyrm/templates/settings/manage_invites.html:3
+msgid "Invites"
+msgstr "Einladungen"
+
+#: bookwyrm/templates/layout.html:118
+msgid "Site Configuration"
+msgstr "Seiteneinstellungen"
+
+#: bookwyrm/templates/layout.html:125
+msgid "Log out"
+msgstr "Abmelden"
+
+#: bookwyrm/templates/layout.html:133 bookwyrm/templates/layout.html:134
+#: bookwyrm/templates/notifications.html:6
+#: bookwyrm/templates/notifications.html:10
+msgid "Notifications"
+msgstr "Benachrichtigungen"
+
+#: bookwyrm/templates/layout.html:151 bookwyrm/templates/layout.html:155
+#: bookwyrm/templates/login.html:17
+#: bookwyrm/templates/snippets/register_form.html:4
+msgid "Username:"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:156
+msgid "password"
+msgstr "Passwort"
+
+#: bookwyrm/templates/layout.html:157 bookwyrm/templates/login.html:36
+msgid "Forgot your password?"
+msgstr "Passwort vergessen?"
+
+#: bookwyrm/templates/layout.html:160 bookwyrm/templates/login.html:10
+#: bookwyrm/templates/login.html:33
+msgid "Log in"
+msgstr "Anmelden"
+
+#: bookwyrm/templates/layout.html:168
+msgid "Join"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:191
+msgid "About this server"
+msgstr "Über diesen Server"
+
+#: bookwyrm/templates/layout.html:195
+msgid "Contact site admin"
+msgstr "Admin kontaktieren"
+
+#: bookwyrm/templates/layout.html:202
+#, python-format
+msgid "Support %(site_name)s on %(support_title)s "
+msgstr ""
+
+#: bookwyrm/templates/layout.html:206
+msgid "BookWyrm is open source software. You can contribute or report issues on GitHub ."
+msgstr "BookWyrm ist open source Software. Du kannst dich auf GitHub beteiligen oder etwas melden."
+
+#: bookwyrm/templates/lists/create_form.html:5
+#: bookwyrm/templates/lists/lists.html:17
+msgid "Create List"
+msgstr "Liste erstellen"
+
+#: bookwyrm/templates/lists/created_text.html:5
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Created and curated by %(username)s "
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/lists/created_text.html:7
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Created by %(username)s "
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/lists/curate.html:6
+msgid "Pending Books"
+msgstr "Unbestätigte Bücher"
+
+#: bookwyrm/templates/lists/curate.html:7
+msgid "Go to list"
+msgstr "Zur Liste"
+
+#: bookwyrm/templates/lists/curate.html:9
+msgid "You're all set!"
+msgstr "Du bist soweit!"
+
+#: bookwyrm/templates/lists/curate.html:15
+msgid "Suggested by"
+msgstr "Vorgeschlagen von"
+
+#: bookwyrm/templates/lists/curate.html:35
+msgid "Approve"
+msgstr "Bestätigen"
+
+#: bookwyrm/templates/lists/curate.html:41
+msgid "Discard"
+msgstr "Ablehnen"
+
+#: bookwyrm/templates/lists/edit_form.html:5
+#: bookwyrm/templates/lists/list_layout.html:18
+msgid "Edit List"
+msgstr "Liste bearbeiten"
+
+#: bookwyrm/templates/lists/form.html:18
+msgid "List curation:"
+msgstr "Listenkuratierung:"
+
+#: bookwyrm/templates/lists/form.html:21
+msgid "Closed"
+msgstr "Geschlossen"
+
+#: bookwyrm/templates/lists/form.html:22
+msgid "Only you can add and remove books to this list"
+msgstr "Nur du kannst Bücher hinzufügen oder entfernen"
+
+#: bookwyrm/templates/lists/form.html:26
+msgid "Curated"
+msgstr "Kuratiert"
+
+#: bookwyrm/templates/lists/form.html:27
+msgid "Anyone can suggest books, subject to your approval"
+msgstr "Alle können Bücher vorschlagen, du kannst diese bestätigen"
+
+#: bookwyrm/templates/lists/form.html:31
+msgid "Open"
+msgstr "Offen"
+
+#: bookwyrm/templates/lists/form.html:32
+msgid "Anyone can add books to this list"
+msgstr "Alle können Bücher hinzufügen"
+
+#: bookwyrm/templates/lists/list.html:17
+msgid "This list is currently empty"
+msgstr "Diese Liste ist momentan leer"
+
+#: bookwyrm/templates/lists/list.html:35
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Added by %(username)s "
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/lists/list.html:41
+msgid "Remove"
+msgstr "Entfernen"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Add Books"
+msgstr "Bücher hinzufügen"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Suggest Books"
+msgstr "Bücher vorschlagen"
+
+#: bookwyrm/templates/lists/list.html:58
+msgid "Search for a book"
+msgstr "Nach einem Buch suchen"
+
+#: bookwyrm/templates/lists/list.html:63
+msgid "search"
+msgstr "suchen"
+
+#: bookwyrm/templates/lists/list.html:69
+msgid "Clear search"
+msgstr "Suche leeren"
+
+#: bookwyrm/templates/lists/list.html:74
+#, python-format
+msgid "No books found matching the query \"%(query)s\""
+msgstr "Keine passenden Bücher zu \"%(query)s\" gefunden"
+
+#: bookwyrm/templates/lists/list.html:75
+msgid "No books found"
+msgstr "Keine Bücher gefunden"
+
+#: bookwyrm/templates/lists/list.html:89
+msgid "Suggest"
+msgstr "Vorschlagen"
+
+#: bookwyrm/templates/lists/lists.html:14
+msgid "Your lists"
+msgstr "Deine Listen"
+
+#: bookwyrm/templates/lists/lists.html:32
+#, fuzzy, python-format
+#| msgid "See all %(size)s"
+msgid "See all %(size)s lists"
+msgstr "Alle %(size)s anzeigen"
+
+#: bookwyrm/templates/lists/lists.html:40
+msgid "Recent Lists"
+msgstr "Aktuelle Listen"
+
+#: bookwyrm/templates/login.html:4
+msgid "Login"
+msgstr ""
+
+#: bookwyrm/templates/login.html:23 bookwyrm/templates/password_reset.html:17
+#: bookwyrm/templates/snippets/register_form.html:22
+msgid "Password:"
+msgstr "Passwort:"
+
+#: bookwyrm/templates/login.html:49
+msgid "Contact an administrator to get an invite"
+msgstr "Kontaktiere für eine Einladung eine*n Admin"
+
+#: bookwyrm/templates/login.html:59
+msgid "More about this site"
+msgstr "Mehr über diese Seite"
+
+#: bookwyrm/templates/notfound.html:4 bookwyrm/templates/notfound.html:8
+msgid "Not Found"
+msgstr "Nicht gefunden"
+
+#: bookwyrm/templates/notfound.html:9
+msgid "The page you requested doesn't seem to exist!"
+msgstr "Die Seite die du angefordert hast scheint nicht zu existieren!"
+
+#: bookwyrm/templates/notifications.html:14
+msgid "Delete notifications"
+msgstr "Benachrichtigungen löschen"
+
+#: bookwyrm/templates/notifications.html:51
+#, python-format
+msgid "favorited your review of %(book_title)s "
+msgstr "hat deine Bewertung von %(book_title)s favorisiert"
+
+#: bookwyrm/templates/notifications.html:53
+#, python-format
+msgid "favorited your comment on %(book_title)s "
+msgstr "hat deinen Kommentar zu %(book_title)s favorisiert"
+
+#: bookwyrm/templates/notifications.html:55
+#, python-format
+msgid "favorited your quote from %(book_title)s "
+msgstr " hat dein Zitat aus %(book_title)s favorisiert"
+
+#: bookwyrm/templates/notifications.html:57
+#, python-format
+msgid "favorited your status "
+msgstr "hat deinen Status favorisiert"
+
+#: bookwyrm/templates/notifications.html:62
+#, python-format
+msgid "mentioned you in a review of %(book_title)s "
+msgstr "hat dich in einer Bewertung von %(book_title)s erwähnt"
+
+#: bookwyrm/templates/notifications.html:64
+#, python-format
+msgid "mentioned you in a comment on %(book_title)s "
+msgstr "hat dich in einem Kommentar zu %(book_title)s erwähnt"
+
+#: bookwyrm/templates/notifications.html:66
+#, python-format
+msgid "mentioned you in a quote from %(book_title)s "
+msgstr "hat dich in einem Zitat von %(book_title)s erwähnt"
+
+#: bookwyrm/templates/notifications.html:68
+#, python-format
+msgid "mentioned you in a status "
+msgstr "hat dich in einem Status erwähnt"
+
+#: bookwyrm/templates/notifications.html:73
+#, python-format
+msgid "replied to your review of %(book_title)s "
+msgstr "hat auf deine Bewertung von %(book_title)s geantwortet "
+
+#: bookwyrm/templates/notifications.html:75
+#, python-format
+msgid "replied to your comment on %(book_title)s "
+msgstr "hat auf deinen Kommentar zu %(book_title)s geantwortet "
+
+#: bookwyrm/templates/notifications.html:77
+#, python-format
+msgid "replied to your quote from %(book_title)s "
+msgstr "hat auf dein Zitat aus %(book_title)s geantwortet "
+
+#: bookwyrm/templates/notifications.html:79
+#, python-format
+msgid "replied to your status "
+msgstr "hat auf deinen Status geantwortet "
+
+#: bookwyrm/templates/notifications.html:83
+msgid "followed you"
+msgstr "folgt dir"
+
+#: bookwyrm/templates/notifications.html:86
+msgid "sent you a follow request"
+msgstr "hat dir eine Folgeanfrage geschickt"
+
+#: bookwyrm/templates/notifications.html:92
+#, python-format
+msgid "boosted your review of %(book_title)s "
+msgstr "hat deine Bewertung von %(book_title)s geteilt"
+
+#: bookwyrm/templates/notifications.html:94
+#, python-format
+msgid "boosted your comment on%(book_title)s "
+msgstr "hat deinen Kommentar zu%(book_title)s geteilt"
+
+#: bookwyrm/templates/notifications.html:96
+#, python-format
+msgid "boosted your quote from %(book_title)s "
+msgstr "hat dein Zitat aus %(book_title)s geteilt"
+
+#: bookwyrm/templates/notifications.html:98
+#, python-format
+msgid "boosted your status "
+msgstr "hat deinen Status geteilt"
+
+#: bookwyrm/templates/notifications.html:102
+#, python-format
+msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgstr "hat %(book_title)s zu deiner Liste \"%(list_name)s \" Hinzugefügt"
+
+#: bookwyrm/templates/notifications.html:104
+#, python-format
+msgid " suggested adding %(book_title)s to your list \"%(list_name)s \""
+msgstr "hat %(book_title)s für deine Liste \"%(list_name)s \" vorgeschlagen"
+
+#: bookwyrm/templates/notifications.html:108
+#, python-format
+msgid " your import completed."
+msgstr " dein Import ist abgeschlossen."
+
+#: bookwyrm/templates/notifications.html:142
+msgid "You're all caught up!"
+msgstr "Du bist auf dem neusten Stand!"
+
+#: bookwyrm/templates/password_reset.html:4
+#: bookwyrm/templates/password_reset.html:10
+#: bookwyrm/templates/password_reset_request.html:4
+#: bookwyrm/templates/password_reset_request.html:10
+msgid "Reset Password"
+msgstr "Passwort zurücksetzen!"
+
+#: bookwyrm/templates/password_reset.html:23
+#: bookwyrm/templates/preferences/change_password.html:18
+msgid "Confirm password:"
+msgstr "Passwort bestätigen:"
+
+#: bookwyrm/templates/password_reset.html:30
+msgid "Confirm"
+msgstr "Bestätigen"
+
+#: bookwyrm/templates/password_reset_request.html:12
+msgid "A link to reset your password will be sent to your email address"
+msgstr "Ein Link zum Zurücksetzen deines Passworts wird an deine Mailadresse geschickt"
+
+#: bookwyrm/templates/password_reset_request.html:16
+#: bookwyrm/templates/preferences/edit_user.html:38
+#: bookwyrm/templates/snippets/register_form.html:13
+msgid "Email address:"
+msgstr "E-Mail Adresse"
+
+#: bookwyrm/templates/password_reset_request.html:23
+msgid "Reset password"
+msgstr "Passwort zurücksetzen"
+
+#: bookwyrm/templates/preferences/blocks.html:4
+#: bookwyrm/templates/preferences/blocks.html:7
+#: bookwyrm/templates/preferences/preferences_layout.html:23
+msgid "Blocked Users"
+msgstr "Blockierte Nutzer*innen"
+
+#: bookwyrm/templates/preferences/blocks.html:12
+msgid "No users currently blocked."
+msgstr "Momentan keine Nutzer*innen blockiert."
+
+#: bookwyrm/templates/preferences/change_password.html:4
+#: bookwyrm/templates/preferences/change_password.html:7
+#: bookwyrm/templates/preferences/change_password.html:21
+#: bookwyrm/templates/preferences/preferences_layout.html:17
+msgid "Change Password"
+msgstr "Passwort ändern"
+
+#: bookwyrm/templates/preferences/change_password.html:14
+msgid "New password:"
+msgstr "Neues Passwort:"
+
+#: bookwyrm/templates/preferences/edit_user.html:4
+#: bookwyrm/templates/preferences/edit_user.html:7
+msgid "Edit Profile"
+msgstr "Profil bearbeiten:"
+
+#: bookwyrm/templates/preferences/edit_user.html:17
+msgid "Avatar:"
+msgstr ""
+
+#: bookwyrm/templates/preferences/edit_user.html:24
+msgid "Display name:"
+msgstr "Displayname:"
+
+#: bookwyrm/templates/preferences/edit_user.html:31
+msgid "Summary:"
+msgstr "Bio:"
+
+#: bookwyrm/templates/preferences/edit_user.html:46
+msgid "Manually approve followers:"
+msgstr "Folgende manuell bestätigen"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:11
+msgid "Account"
+msgstr ""
+
+#: bookwyrm/templates/preferences/preferences_layout.html:20
+msgid "Relationships"
+msgstr "Beziehungen"
+
+#: bookwyrm/templates/search_results.html:33
+msgid "Didn't find what you were looking for?"
+msgstr "Nicht gefunden, wonach du gesucht hast?"
+
+#: bookwyrm/templates/search_results.html:35
+msgid "Show results from other catalogues"
+msgstr "Ergebnisse aus anderen Katalogen zeigen"
+
+#: bookwyrm/templates/search_results.html:57
+msgid "Import book"
+msgstr "Buch importieren"
+
+#: bookwyrm/templates/search_results.html:67
+msgid "Hide results from other catalogues"
+msgstr "Ergebnisse aus anderen Katalogen ausblenden"
+
+#: bookwyrm/templates/search_results.html:75
+msgid "Matching Users"
+msgstr "Passende Nutzer*innen"
+
+#: bookwyrm/templates/search_results.html:77
+#, python-format
+msgid "No users found for \"%(query)s\""
+msgstr "Keine Nutzer*innen für \"%(query)s\" gefunden"
+
+#: bookwyrm/templates/search_results.html:94
+#, python-format
+msgid "No lists found for \"%(query)s\""
+msgstr "Keine Liste für \"%(query)s\" gefunden"
+
+#: bookwyrm/templates/settings/admin_layout.html:4
+msgid "Administration"
+msgstr ""
+
+#: bookwyrm/templates/settings/admin_layout.html:15
+msgid "Manage Users"
+msgstr "Nutzer*innen verwalten"
+
+#: bookwyrm/templates/settings/admin_layout.html:23
+#: bookwyrm/templates/settings/federation.html:4
+msgid "Federated Servers"
+msgstr "Föderierende Server"
+
+#: bookwyrm/templates/settings/admin_layout.html:28
+msgid "Instance Settings"
+msgstr "Instanzeinstellungen"
+
+#: bookwyrm/templates/settings/admin_layout.html:32
+#: bookwyrm/templates/settings/site.html:4
+#: bookwyrm/templates/settings/site.html:6
+msgid "Site Settings"
+msgstr "Seiteneinstellungen"
+
+#: bookwyrm/templates/settings/admin_layout.html:35
+#: bookwyrm/templates/settings/site.html:13
+msgid "Instance Info"
+msgstr "Instanzinformationen"
+
+#: bookwyrm/templates/settings/admin_layout.html:36
+#: bookwyrm/templates/settings/site.html:39
+msgid "Images"
+msgstr "Bilder"
+
+#: bookwyrm/templates/settings/admin_layout.html:37
+#: bookwyrm/templates/settings/site.html:59
+msgid "Footer Content"
+msgstr "Inhalt des Footers"
+
+#: bookwyrm/templates/settings/admin_layout.html:38
+#: bookwyrm/templates/settings/site.html:77
+msgid "Registration"
+msgstr "Registrierung"
+
+#: bookwyrm/templates/settings/federation.html:10
+msgid "Server name"
+msgstr "Servername"
+
+#: bookwyrm/templates/settings/federation.html:11
+msgid "Software"
+msgstr ""
+
+#: bookwyrm/templates/settings/federation.html:12
+msgid "Status"
+msgstr ""
+
+#: bookwyrm/templates/settings/manage_invites.html:7
+msgid "Generate New Invite"
+msgstr "Neue Einladung erzeugen"
+
+#: bookwyrm/templates/settings/manage_invites.html:13
+msgid "Expiry:"
+msgstr "Ablaufen:"
+
+#: bookwyrm/templates/settings/manage_invites.html:19
+msgid "Use limit:"
+msgstr "Begrenzte Benutzung"
+
+#: bookwyrm/templates/settings/manage_invites.html:26
+msgid "Create Invite"
+msgstr "Einladung erstellen"
+
+#: bookwyrm/templates/settings/manage_invites.html:33
+msgid "Link"
+msgstr ""
+
+#: bookwyrm/templates/settings/manage_invites.html:34
+msgid "Expires"
+msgstr "Läuft aus"
+
+#: bookwyrm/templates/settings/manage_invites.html:35
+msgid "Max uses"
+msgstr "Maximale Benutzungen"
+
+#: bookwyrm/templates/settings/manage_invites.html:36
+msgid "Times used"
+msgstr "Mal benutzt"
+
+#: bookwyrm/templates/settings/manage_invites.html:39
+msgid "No active invites"
+msgstr "Keine aktiven Einladungen"
+
+#: bookwyrm/templates/settings/site.html:15
+msgid "Instance Name:"
+msgstr "Instanzname"
+
+#: bookwyrm/templates/settings/site.html:19
+msgid "Tagline:"
+msgstr ""
+
+#: bookwyrm/templates/settings/site.html:23
+msgid "Instance description:"
+msgstr "Instanzbeschreibung"
+
+#: bookwyrm/templates/settings/site.html:27
+msgid "Code of conduct:"
+msgstr ""
+
+#: bookwyrm/templates/settings/site.html:31
+msgid "Privacy Policy:"
+msgstr "Datenschutzerklärung"
+
+#: bookwyrm/templates/settings/site.html:42
+msgid "Logo:"
+msgstr ""
+
+#: bookwyrm/templates/settings/site.html:46
+msgid "Logo small:"
+msgstr "Logo klein"
+
+#: bookwyrm/templates/settings/site.html:50
+msgid "Favicon:"
+msgstr ""
+
+#: bookwyrm/templates/settings/site.html:61
+msgid "Support link:"
+msgstr "Unterstützungslink"
+
+#: bookwyrm/templates/settings/site.html:65
+msgid "Support title:"
+msgstr "Unterstützungstitel"
+
+#: bookwyrm/templates/settings/site.html:69
+msgid "Admin email:"
+msgstr ""
+
+#: bookwyrm/templates/settings/site.html:79
+msgid "Allow registration:"
+msgstr "Registrierungen erlauben"
+
+#: bookwyrm/templates/settings/site.html:83
+msgid "Registration closed text:"
+msgstr "Registrierungen geschlossen text"
+
+#: bookwyrm/templates/snippets/block_button.html:5
+msgid "Block"
+msgstr ""
+
+#: bookwyrm/templates/snippets/block_button.html:10
+msgid "Un-block"
+msgstr ""
+
+#: bookwyrm/templates/snippets/book_titleby.html:3
+#, python-format
+msgid "%(title)s by "
+msgstr "%(title)s von "
+
+#: bookwyrm/templates/snippets/boost_button.html:8
+#: bookwyrm/templates/snippets/boost_button.html:9
+#: bookwyrm/templates/snippets/status/status_body.html:41
+#: bookwyrm/templates/snippets/status/status_body.html:42
+msgid "Boost status"
+msgstr "Status teilen"
+
+#: bookwyrm/templates/snippets/boost_button.html:16
+#: bookwyrm/templates/snippets/boost_button.html:17
+msgid "Un-boost status"
+msgstr "Teilen zurücknehmen"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:3
+msgid "Spoiler alert:"
+msgstr "Spoileralarm:"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:4
+msgid "Spoilers ahead!"
+msgstr "Spoileralarm!"
+
+#: bookwyrm/templates/snippets/create_status.html:9
+msgid "Review"
+msgstr "Bewerten"
+
+#: bookwyrm/templates/snippets/create_status.html:12
+#: bookwyrm/templates/snippets/create_status_form.html:44
+msgid "Comment"
+msgstr "Kommentieren"
+
+#: bookwyrm/templates/snippets/create_status.html:15
+msgid "Quote"
+msgstr "Zitieren"
+
+#: bookwyrm/templates/snippets/create_status_form.html:21
+#: bookwyrm/templates/snippets/shelf.html:17
+msgid "Rating"
+msgstr ""
+
+#: bookwyrm/templates/snippets/create_status_form.html:23
+#: bookwyrm/templates/snippets/rate_action.html:14
+#: bookwyrm/templates/snippets/stars.html:3
+msgid "No rating"
+msgstr "Kein Rating"
+
+#: bookwyrm/templates/snippets/create_status_form.html:54
+msgid "Include spoiler alert"
+msgstr "Spoileralarm aktivieren"
+
+#: bookwyrm/templates/snippets/create_status_form.html:60
+#: bookwyrm/templates/snippets/privacy-icons.html:15
+#: bookwyrm/templates/snippets/privacy-icons.html:16
+#: bookwyrm/templates/snippets/privacy_select.html:19
+msgid "Private"
+msgstr "Privat"
+
+#: bookwyrm/templates/snippets/create_status_form.html:67
+msgid "Post"
+msgstr "Absenden"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:4
+msgid "Delete these read dates?"
+msgstr "Diese Lesedaten löschen?"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:7
+#, python-format
+msgid "You are deleting this readthrough and its %(count)s associated progress updates."
+msgstr "Du löscht diesen Leseforschritt und %(count)s zugehörige Fortschrittsupdates."
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:15
+#: bookwyrm/templates/snippets/follow_request_buttons.html:13
+msgid "Delete"
+msgstr "Löschen"
+
+#: bookwyrm/templates/snippets/fav_button.html:7
+#: bookwyrm/templates/snippets/fav_button.html:8
+#: bookwyrm/templates/snippets/status/status_body.html:45
+#: bookwyrm/templates/snippets/status/status_body.html:46
+msgid "Like status"
+msgstr "Status favorisieren"
+
+#: bookwyrm/templates/snippets/fav_button.html:15
+#: bookwyrm/templates/snippets/fav_button.html:16
+msgid "Un-like status"
+msgstr "Favorisieren zurücknehmen"
+
+#: bookwyrm/templates/snippets/follow_button.html:6
+msgid "Follow request already sent."
+msgstr "Folgeanfrage bereits gesendet."
+
+#: bookwyrm/templates/snippets/follow_button.html:19
+msgid "Send follow request"
+msgstr "Folgeanfrage senden"
+
+#: bookwyrm/templates/snippets/follow_button.html:21
+msgid "Follow"
+msgstr "Folgen"
+
+#: bookwyrm/templates/snippets/follow_button.html:27
+msgid "Unfollow"
+msgstr "Entfolgen"
+
+#: bookwyrm/templates/snippets/follow_request_buttons.html:8
+msgid "Accept"
+msgstr "Annehmen"
+
+#: bookwyrm/templates/snippets/generated_status/goal.html:1
+#, python-format
+msgid "set a goal to read %(counter)s book in %(year)s"
+msgid_plural "set a goal to read %(counter)s books in %(year)s"
+msgstr[0] "Setze das Ziel, %(year)s %(counter)s Buch zu lesen"
+msgstr[1] "Setze das Ziel, %(year)s %(counter)s Bücher zu lesen"
+
+#: bookwyrm/templates/snippets/goal_card.html:21
+msgid "Dismiss message"
+msgstr "Nachricht verwerfen"
+
+#: bookwyrm/templates/snippets/goal_card.html:22
+#, python-format
+msgid "You can set or change your reading goal any time from your profile page "
+msgstr "Du kannst dein Leseziel jederzeit auf deiner Profilseite setzen oder ändern."
+
+#: bookwyrm/templates/snippets/goal_form.html:9
+msgid "Reading goal:"
+msgstr "Leseziel:"
+
+#: bookwyrm/templates/snippets/goal_form.html:14
+msgid "books"
+msgstr "Bücher"
+
+#: bookwyrm/templates/snippets/goal_form.html:19
+msgid "Goal privacy:"
+msgstr "Sichtbarkeit des Ziels"
+
+#: bookwyrm/templates/snippets/goal_form.html:26
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:37
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:29
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:20
+msgid "Post to feed"
+msgstr "Posten"
+
+#: bookwyrm/templates/snippets/goal_form.html:30
+msgid "Set goal"
+msgstr "Ziel setzen"
+
+#: bookwyrm/templates/snippets/goal_progress.html:5
+msgid "Success!"
+msgstr "Erfolg!"
+
+#: bookwyrm/templates/snippets/goal_progress.html:7
+#, python-format
+msgid "%(percent)s%% complete!"
+msgstr "%(percent)s%% komplett!"
+
+#: bookwyrm/templates/snippets/goal_progress.html:10
+#, python-format
+msgid "You've read %(read_count)s of %(goal_count)s books ."
+msgstr "Du hast %(read_count)s von %(goal_count)s Büchern gelesen."
+
+#: bookwyrm/templates/snippets/goal_progress.html:12
+#, python-format
+msgid "%(username)s has read %(read_count)s of %(goal_count)s books ."
+msgstr "%(username)s hat %(read_count)s von %(goal_count)s Büchern gelesen."
+
+#: bookwyrm/templates/snippets/pagination.html:7
+msgid "Previous"
+msgstr "Zurück"
+
+#: bookwyrm/templates/snippets/pagination.html:15
+msgid "Next"
+msgstr "Weiter"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:3
+#: bookwyrm/templates/snippets/privacy-icons.html:4
+#: bookwyrm/templates/snippets/privacy_select.html:10
+msgid "Public"
+msgstr "Öffentlich"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:7
+#: bookwyrm/templates/snippets/privacy-icons.html:8
+#: bookwyrm/templates/snippets/privacy_select.html:13
+msgid "Unlisted"
+msgstr "Ungelistet"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:12
+msgid "Followers-only"
+msgstr "Nur für Folgende"
+
+#: bookwyrm/templates/snippets/privacy_select.html:6
+msgid "Post privacy"
+msgstr ""
+
+#: bookwyrm/templates/snippets/privacy_select.html:16
+#: bookwyrm/templates/user/followers.html:13
+msgid "Followers"
+msgstr "Folgende"
+
+#: bookwyrm/templates/snippets/progress_update.html:6
+msgid "Progress:"
+msgstr "Fortschritt:"
+
+#: bookwyrm/templates/snippets/progress_update.html:16
+#: bookwyrm/templates/snippets/readthrough_form.html:22
+msgid "pages"
+msgstr "Seiten"
+
+#: bookwyrm/templates/snippets/progress_update.html:17
+#: bookwyrm/templates/snippets/readthrough_form.html:23
+msgid "percent"
+msgstr "Prozent"
+
+#: bookwyrm/templates/snippets/progress_update.html:25
+#, python-format
+msgid "of %(book.pages)s pages"
+msgstr "von %(book.pages)s Seiten"
+
+#: bookwyrm/templates/snippets/rate_action.html:4
+msgid "Leave a rating"
+msgstr "Raten"
+
+#: bookwyrm/templates/snippets/rate_action.html:29
+msgid "Rate"
+msgstr ""
+
+#: bookwyrm/templates/snippets/readthrough.html:7
+msgid "Progress Updates:"
+msgstr "Fortschrittsupdates:"
+
+#: bookwyrm/templates/snippets/readthrough.html:11
+msgid "finished"
+msgstr "Abgeschlossen"
+
+#: bookwyrm/templates/snippets/readthrough.html:14
+msgid "Show all updates"
+msgstr "Zeige alle Updates"
+
+#: bookwyrm/templates/snippets/readthrough.html:30
+msgid "Delete this progress update"
+msgstr "Dieses Fortschrittsupdate löschen"
+
+#: bookwyrm/templates/snippets/readthrough.html:40
+msgid "started"
+msgstr "Angefangen"
+
+#: bookwyrm/templates/snippets/readthrough.html:46
+#: bookwyrm/templates/snippets/readthrough.html:60
+msgid "Edit read dates"
+msgstr "Lesedaten bearbeiten"
+
+#: bookwyrm/templates/snippets/readthrough.html:50
+msgid "Delete these read dates"
+msgstr "Diese Lesedaten löschen"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:7
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:19
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:17
+msgid "Started reading"
+msgstr "Zu lesen angefangen"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:14
+msgid "Progress"
+msgstr "Fortschritt"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:30
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:25
+msgid "Finished reading"
+msgstr "Lesen abgeschlossen"
+
+#: bookwyrm/templates/snippets/register_form.html:32
+msgid "Sign Up"
+msgstr "Registrieren"
+
+#: bookwyrm/templates/snippets/rss_title.html:5
+#: bookwyrm/templates/snippets/status/status_header.html:11
+msgid "rated"
+msgstr ""
+
+#: bookwyrm/templates/snippets/rss_title.html:7
+#: bookwyrm/templates/snippets/status/status_header.html:13
+msgid "reviewed"
+msgstr "bewertete"
+
+#: bookwyrm/templates/snippets/rss_title.html:9
+#: bookwyrm/templates/snippets/status/status_header.html:15
+msgid "commented on"
+msgstr "kommentierte"
+
+#: bookwyrm/templates/snippets/rss_title.html:11
+#: bookwyrm/templates/snippets/status/status_header.html:17
+msgid "quoted"
+msgstr "zitierte"
+
+#: bookwyrm/templates/snippets/search_result_text.html:3
+#, python-format
+msgid "by %(author)s"
+msgstr "von %(author)s"
+
+#: bookwyrm/templates/snippets/shelf.html:12
+msgid "Published"
+msgstr "Veröffentlicht"
+
+#: bookwyrm/templates/snippets/shelf.html:13
+msgid "Shelved"
+msgstr "Ins Regal gestellt"
+
+#: bookwyrm/templates/snippets/shelf.html:14
+msgid "Started"
+msgstr "Gestartet"
+
+#: bookwyrm/templates/snippets/shelf.html:15
+msgid "Finished"
+msgstr "Abgeschlossen"
+
+#: bookwyrm/templates/snippets/shelf.html:16
+msgid "External links"
+msgstr "Eterne Links"
+
+#: bookwyrm/templates/snippets/shelf.html:44
+msgid "OpenLibrary"
+msgstr ""
+
+#: bookwyrm/templates/snippets/shelf.html:61
+msgid "This shelf is empty."
+msgstr "Dieses Regal ist leer."
+
+#: bookwyrm/templates/snippets/shelf.html:67
+msgid "Delete shelf"
+msgstr "Regal löschen"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:4
+msgid "Change shelf"
+msgstr "Regal wechseln"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:27
+msgid "Unshelve"
+msgstr "Vom Regal nehmen"
+
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:5
+#, python-format
+msgid "Finish \"%(book_title)s \""
+msgstr "\"%(book_title)s \" abschließen"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_dropdown.html:5
+msgid "More shelves"
+msgstr "Mehr Regale"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:8
+msgid "Start reading"
+msgstr "Zu lesen beginnen"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:13
+msgid "Finish reading"
+msgstr "Lesen abschließen"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:16
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
+msgid "Want to read"
+msgstr "Auf Leseliste setzen"
+
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:5
+#, python-format
+msgid "Start \"%(book_title)s \""
+msgstr "\"%(book_title)s \" beginnen"
+
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:5
+#, python-format
+msgid "Want to Read \"%(book_title)s \""
+msgstr "\"%(book_title)s \" auf Leseliste setzen"
+
+#: bookwyrm/templates/snippets/status/status.html:9
+msgid "boosted"
+msgstr "teilt"
+
+#: bookwyrm/templates/snippets/status/status_body.html:24
+#: bookwyrm/templates/snippets/status/status_body.html:37
+#: bookwyrm/templates/snippets/status/status_body.html:38
+msgid "Reply"
+msgstr "Antwort"
+
+#: bookwyrm/templates/snippets/status/status_content.html:18
+#: bookwyrm/templates/snippets/trimmed_text.html:15
+msgid "Show more"
+msgstr "Mehr anzeigen"
+
+#: bookwyrm/templates/snippets/status/status_content.html:25
+#: bookwyrm/templates/snippets/trimmed_text.html:25
+msgid "Show less"
+msgstr "Weniger anzeigen"
+
+#: bookwyrm/templates/snippets/status/status_content.html:46
+msgid "Open image in new window"
+msgstr "Bild in neuem Fenster öffnen"
+
+#: bookwyrm/templates/snippets/status/status_header.html:22
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "replied to %(username)s's review "
+msgstr "Direktnachrichten mit %(username)s "
+
+#: bookwyrm/templates/snippets/status/status_header.html:24
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's comment "
+msgstr "hat auf deinen Status geantwortet"
+
+#: bookwyrm/templates/snippets/status/status_header.html:26
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's quote "
+msgstr "hat auf deinen Status geantwortet "
+
+#: bookwyrm/templates/snippets/status/status_header.html:28
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's status "
+msgstr "hat auf deinen Status geantwortet "
+
+#: bookwyrm/templates/snippets/status/status_options.html:7
+#: bookwyrm/templates/snippets/user_options.html:7
+msgid "More options"
+msgstr "Mehr Optionen"
+
+#: bookwyrm/templates/snippets/status/status_options.html:17
+msgid "Delete status"
+msgstr "Post löschen"
+
+#: bookwyrm/templates/snippets/status/status_options.html:23
+#: bookwyrm/templates/snippets/user_options.html:13
+msgid "Send direct message"
+msgstr "Direktnachricht senden"
+
+#: bookwyrm/templates/snippets/switch_edition_button.html:5
+msgid "Switch to this edition"
+msgstr "Zu dieser Edition wechseln"
+
+#: bookwyrm/templates/snippets/tag.html:14
+msgid "Remove tag"
+msgstr "Tag entfernen"
+
+#: bookwyrm/templates/tag.html:9
+#, python-format
+msgid "Books tagged \"%(tag.name)s\""
+msgstr "Mit \"%(tag.name)s\" markierte Bücher"
+
+#: bookwyrm/templates/user/create_shelf_form.html:5
+#: bookwyrm/templates/user/create_shelf_form.html:22
+msgid "Create Shelf"
+msgstr "Regal erstellen"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:5
+msgid "Edit Shelf"
+msgstr "Regal bearbeiten"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:26
+msgid "Update shelf"
+msgstr "Regal aktualisieren"
+
+#: bookwyrm/templates/user/followers.html:7
+#: bookwyrm/templates/user/following.html:7 bookwyrm/templates/user/user.html:9
+msgid "User Profile"
+msgstr "Benutzerprofil"
+
+#: bookwyrm/templates/user/followers.html:29
+#, python-format
+msgid "%(username)s has no followers"
+msgstr "niemand folgt %(username)s "
+
+#: bookwyrm/templates/user/following.html:13
+msgid "Following"
+msgstr "Folgend"
+
+#: bookwyrm/templates/user/following.html:29
+#, python-format
+msgid "%(username)s isn't following any users"
+msgstr "%(username)s folgt niemandem"
+
+#: bookwyrm/templates/user/lists.html:9
+msgid "Your Lists"
+msgstr "Deine Listen"
+
+#: bookwyrm/templates/user/lists.html:11
+#, python-format
+msgid "Lists: %(username)s"
+msgstr "Listen: %(username)s"
+
+#: bookwyrm/templates/user/lists.html:17 bookwyrm/templates/user/lists.html:29
+msgid "Create list"
+msgstr "Liste Erstellen"
+
+#: bookwyrm/templates/user/shelf.html:9
+msgid "Your Shelves"
+msgstr "Deine Regale"
+
+#: bookwyrm/templates/user/shelf.html:11
+#, python-format
+msgid "%(username)s: Shelves"
+msgstr "%(username)s: Regale"
+
+#: bookwyrm/templates/user/shelf.html:33
+msgid "Create shelf"
+msgstr "Regal erstellen"
+
+#: bookwyrm/templates/user/shelf.html:54
+msgid "Edit shelf"
+msgstr "Regal bearbeiten"
+
+#: bookwyrm/templates/user/user.html:15
+msgid "Edit profile"
+msgstr "Profil bearbeiten"
+
+#: bookwyrm/templates/user/user.html:26
+#: bookwyrm/templates/user/user_layout.html:68
+msgid "Shelves"
+msgstr "Regale"
+
+#: bookwyrm/templates/user/user.html:31
+#, python-format
+msgid "See all %(size)s"
+msgstr "Alle %(size)s anzeigen"
+
+#: bookwyrm/templates/user/user.html:44
+#, python-format
+msgid "See all %(shelf_count)s shelves"
+msgstr "Alle %(shelf_count)s Regale anzeigen"
+
+#: bookwyrm/templates/user/user.html:56
+#, python-format
+msgid "Set a reading goal for %(year)s"
+msgstr "Leseziel für %(year)s setzen"
+
+#: bookwyrm/templates/user/user.html:62
+msgid "User Activity"
+msgstr "Nutzer*innenaktivität"
+
+#: bookwyrm/templates/user/user.html:65
+msgid "RSS feed"
+msgstr ""
+
+#: bookwyrm/templates/user/user.html:76
+msgid "No activities yet!"
+msgstr "Noch keine Aktivitäten!"
+
+#: bookwyrm/templates/user/user_layout.html:32
+msgid "Follow Requests"
+msgstr "Folgeanfragen"
+
+#: bookwyrm/templates/user/user_layout.html:50
+msgid "Activity"
+msgstr "Aktivität"
+
+#: bookwyrm/templates/user/user_layout.html:56
+msgid "Reading Goal"
+msgstr "Leseziel"
+
+#: bookwyrm/templates/user/user_preview.html:13
+#, python-format
+msgid "Joined %(date)s"
+msgstr "Beigetreten %(date)s"
+
+#: bookwyrm/templates/user/user_preview.html:15
+#, python-format
+msgid "%(counter)s follower"
+msgid_plural "%(counter)s followers"
+msgstr[0] "%(counter)s Folgende*r"
+msgstr[1] "%(counter)s Folgende"
+
+#: bookwyrm/templates/user/user_preview.html:16
+#, python-format
+msgid "%(counter)s following"
+msgstr "Folgt %(counter)s"
+
+#~ msgid "Created and curated by"
+#~ msgstr "Erstellt und kuratiert von"
+
+#~ msgid "Created by"
+#~ msgstr "Erstellt von"
+
+#~ msgid "Added by"
+#~ msgstr "Hinzugefügt von"
+
+#~ msgid "Create New Shelf"
+#~ msgstr "Neues Regal erstellen"
+
+#~ msgid "Create new list"
+#~ msgstr "Neue Liste erstellen"
diff --git a/locale/en_US/LC_MESSAGES/django.mo b/locale/en_US/LC_MESSAGES/django.mo
index c0a5dd97..e0752914 100644
Binary files a/locale/en_US/LC_MESSAGES/django.mo and b/locale/en_US/LC_MESSAGES/django.mo differ
diff --git a/locale/en_US/LC_MESSAGES/django.po b/locale/en_US/LC_MESSAGES/django.po
index 4d140e52..d69fe702 100644
--- a/locale/en_US/LC_MESSAGES/django.po
+++ b/locale/en_US/LC_MESSAGES/django.po
@@ -1,119 +1,251 @@
-# English language text for the bookwyrm UI
+# Stub English-language trnaslation file
# Copyright (C) 2021 Mouse Reeve
-# This file is distributed under the same license as the bookwyrm package.
+# This file is distributed under the same license as the BookWyrm package.
# Mouse Reeve , 2021
#
#, fuzzy
msgid ""
msgstr ""
-"Project-Id-Version: 0.1.1\n"
+"Project-Id-Version: 0.0.1\n"
"Report-Msgid-Bugs-To: \n"
-"POT-Creation-Date: 2021-02-28 08:42-0800\n"
-"PO-Revision-Date: 2021-02-27 13:50+PST\n"
+"POT-Creation-Date: 2021-03-07 23:40+0000\n"
+"PO-Revision-Date: 2021-02-28 17:19-0800\n"
"Last-Translator: Mouse Reeve \n"
-"Language-Team: Mouse Reeve \n"
-"Language: English \n"
+"Language-Team: English \n"
+"Language: English\n"
"MIME-Version: 1.0\n"
"Content-Type: text/plain; charset=UTF-8\n"
"Content-Transfer-Encoding: 8bit\n"
"Plural-Forms: nplurals=2; plural=(n != 1);\n"
-#: bookwyrm/templates/author.html:13 bookwyrm/templates/author.html:14
+#: bookwyrm/forms.py:185
+msgid "One Day"
+msgstr ""
+
+#: bookwyrm/forms.py:186
+msgid "One Week"
+msgstr ""
+
+#: bookwyrm/forms.py:187
+msgid "One Month"
+msgstr ""
+
+#: bookwyrm/forms.py:188
+msgid "Does Not Expire"
+msgstr ""
+
+#: bookwyrm/forms.py:190
+#, python-format
+msgid "%(count)d uses"
+msgstr ""
+
+#: bookwyrm/forms.py:192
+msgid "Unlimited"
+msgstr ""
+
+#: bookwyrm/models/fields.py:24
+#, python-format
+msgid "%(value)s is not a valid remote_id"
+msgstr ""
+
+#: bookwyrm/models/fields.py:33 bookwyrm/models/fields.py:42
+#, python-format
+msgid "%(value)s is not a valid username"
+msgstr ""
+
+#: bookwyrm/models/fields.py:164 bookwyrm/templates/layout.html:152
+msgid "username"
+msgstr ""
+
+#: bookwyrm/models/fields.py:169
+msgid "A user with that username already exists."
+msgstr ""
+
+#: bookwyrm/settings.py:142
+msgid "English"
+msgstr ""
+
+#: bookwyrm/settings.py:143
+msgid "German"
+msgstr ""
+
+#: bookwyrm/settings.py:144
+msgid "Spanish"
+msgstr ""
+
+#: bookwyrm/settings.py:145
+msgid "French"
+msgstr ""
+
+#: bookwyrm/settings.py:146
+msgid "Simplified Chinese"
+msgstr ""
+
+#: bookwyrm/templates/author.html:16 bookwyrm/templates/author.html:17
+#: bookwyrm/templates/edit_author.html:5
msgid "Edit Author"
msgstr ""
-#: bookwyrm/templates/author.html:29
+#: bookwyrm/templates/author.html:32
msgid "Wikipedia"
msgstr ""
-#: bookwyrm/templates/author.html:34
+#: bookwyrm/templates/author.html:37
#, python-format
msgid "Books by %(name)s"
msgstr ""
-#: bookwyrm/templates/book.html:27 bookwyrm/templates/book.html:28
+#: bookwyrm/templates/book.html:21
+#: bookwyrm/templates/discover/large-book.html:12
+#: bookwyrm/templates/discover/small-book.html:9
+msgid "by"
+msgstr ""
+
+#: bookwyrm/templates/book.html:29 bookwyrm/templates/book.html:30
+#: bookwyrm/templates/edit_book.html:5
msgid "Edit Book"
msgstr ""
-#: bookwyrm/templates/book.html:43
+#: bookwyrm/templates/book.html:45
msgid "Add cover"
msgstr ""
-#: bookwyrm/templates/book.html:49 bookwyrm/templates/lists/list.html:89
+#: bookwyrm/templates/book.html:51 bookwyrm/templates/lists/list.html:89
msgid "Add"
msgstr ""
-#: bookwyrm/templates/book.html:58
+#: bookwyrm/templates/book.html:60
msgid "ISBN:"
msgstr ""
-#: bookwyrm/templates/book.html:65 bookwyrm/templates/edit_book.html:104
+#: bookwyrm/templates/book.html:67 bookwyrm/templates/edit_book.html:107
msgid "OCLC Number:"
msgstr ""
-#: bookwyrm/templates/book.html:72 bookwyrm/templates/edit_book.html:108
+#: bookwyrm/templates/book.html:74 bookwyrm/templates/edit_book.html:111
msgid "ASIN:"
msgstr ""
#: bookwyrm/templates/book.html:84
+#, python-format
+msgid "%(format)s, %(pages)s pages"
+msgstr ""
+
+#: bookwyrm/templates/book.html:86
+#, python-format
+msgid "%(pages)s pages"
+msgstr ""
+
+#: bookwyrm/templates/book.html:91
msgid "View on OpenLibrary"
msgstr ""
-#: bookwyrm/templates/book.html:102 bookwyrm/templates/edit_book.html:36
+#: bookwyrm/templates/book.html:100
+#, python-format
+msgid "(%(review_count)s review)"
+msgid_plural "(%(review_count)s reviews)"
+msgstr[0] ""
+msgstr[1] ""
+
+#: bookwyrm/templates/book.html:106
+msgid "Add Description"
+msgstr ""
+
+#: bookwyrm/templates/book.html:113 bookwyrm/templates/edit_book.html:39
#: bookwyrm/templates/lists/form.html:12
msgid "Description:"
msgstr ""
-#: bookwyrm/templates/book.html:106 bookwyrm/templates/edit_author.html:75
-#: bookwyrm/templates/edit_book.html:117 bookwyrm/templates/lists/form.html:42
-#: bookwyrm/templates/preferences/edit_user.html:47
-#: bookwyrm/templates/settings/site.html:86
+#: bookwyrm/templates/book.html:117 bookwyrm/templates/edit_author.html:78
+#: bookwyrm/templates/edit_book.html:120 bookwyrm/templates/lists/form.html:42
+#: bookwyrm/templates/preferences/edit_user.html:50
+#: bookwyrm/templates/settings/site.html:89
#: bookwyrm/templates/snippets/progress_update.html:21
-#: bookwyrm/templates/snippets/readthrough.html:61
+#: bookwyrm/templates/snippets/readthrough.html:64
#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:42
#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:34
msgid "Save"
msgstr ""
-#: bookwyrm/templates/book.html:138
+#: bookwyrm/templates/book.html:118 bookwyrm/templates/book.html:167
+#: bookwyrm/templates/edit_author.html:79 bookwyrm/templates/edit_book.html:121
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:17
+#: bookwyrm/templates/snippets/goal_form.html:32
+#: bookwyrm/templates/snippets/readthrough.html:65
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:43
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:35
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:28
+msgid "Cancel"
+msgstr ""
+
+#: bookwyrm/templates/book.html:127
+#, python-format
+msgid "%(count)s editions "
+msgstr ""
+
+#: bookwyrm/templates/book.html:135
+#, python-format
+msgid "This edition is on your %(shelf_name)s shelf."
+msgstr ""
+
+#: bookwyrm/templates/book.html:141
+#, python-format
+msgid "A different edition of this book is on your %(shelf_name)s shelf."
+msgstr ""
+
+#: bookwyrm/templates/book.html:150
msgid "Your reading activity"
msgstr ""
-#: bookwyrm/templates/book.html:144
+#: bookwyrm/templates/book.html:152
+msgid "Add read dates"
+msgstr ""
+
+#: bookwyrm/templates/book.html:157
msgid "You don't have any reading activity for this book."
msgstr ""
-#: bookwyrm/templates/book.html:151
+#: bookwyrm/templates/book.html:164
msgid "Create"
msgstr ""
-#: bookwyrm/templates/book.html:172
+#: bookwyrm/templates/book.html:186
msgid "Tags"
msgstr ""
-#: bookwyrm/templates/book.html:176 bookwyrm/templates/snippets/tag.html:18
+#: bookwyrm/templates/book.html:190 bookwyrm/templates/snippets/tag.html:18
msgid "Add tag"
msgstr ""
-#: bookwyrm/templates/book.html:193
+#: bookwyrm/templates/book.html:207
msgid "Subjects"
msgstr ""
-#: bookwyrm/templates/book.html:204
+#: bookwyrm/templates/book.html:218
msgid "Places"
msgstr ""
-#: bookwyrm/templates/book.html:215 bookwyrm/templates/layout.html:64
-#: bookwyrm/templates/lists/lists.html:6
-#: bookwyrm/templates/search_results.html:85
-#: bookwyrm/templates/user/user_layout.html:60
+#: bookwyrm/templates/book.html:229 bookwyrm/templates/layout.html:64
+#: bookwyrm/templates/lists/lists.html:4 bookwyrm/templates/lists/lists.html:9
+#: bookwyrm/templates/search_results.html:92
+#: bookwyrm/templates/user/user_layout.html:62
msgid "Lists"
msgstr ""
-#: bookwyrm/templates/book.html:244
+#: bookwyrm/templates/book.html:258
msgid "rated it"
msgstr ""
+#: bookwyrm/templates/components/inline_form.html:8
+#: bookwyrm/templates/feed/feed_layout.html:54
+msgid "Close"
+msgstr ""
+
+#: bookwyrm/templates/discover/about.html:7
+#, python-format
+msgid "About %(site_name)s"
+msgstr ""
+
#: bookwyrm/templates/discover/about.html:10
#: bookwyrm/templates/discover/about.html:20
msgid "Code of Conduct"
@@ -128,162 +260,180 @@ msgstr ""
msgid "Recent Books"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:15
+#: bookwyrm/templates/discover/landing_layout.html:5
+msgid "Welcome"
+msgstr ""
+
+#: bookwyrm/templates/discover/landing_layout.html:17
msgid "Decentralized"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:21
+#: bookwyrm/templates/discover/landing_layout.html:23
msgid "Friendly"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:27
+#: bookwyrm/templates/discover/landing_layout.html:29
msgid "Anti-Corporate"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:42
+#: bookwyrm/templates/discover/landing_layout.html:44
#, python-format
msgid "Join %(name)s"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:47
-#: bookwyrm/templates/login.html:46
+#: bookwyrm/templates/discover/landing_layout.html:49
+#: bookwyrm/templates/login.html:48
msgid "This instance is closed"
msgstr ""
-#: bookwyrm/templates/discover/landing_layout.html:53
+#: bookwyrm/templates/discover/landing_layout.html:55
msgid "Your Account"
msgstr ""
-#: bookwyrm/templates/edit_author.html:10 bookwyrm/templates/edit_book.html:10
+#: bookwyrm/templates/edit_author.html:13 bookwyrm/templates/edit_book.html:13
msgid "Added:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:11 bookwyrm/templates/edit_book.html:11
+#: bookwyrm/templates/edit_author.html:14 bookwyrm/templates/edit_book.html:14
msgid "Updated:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:12 bookwyrm/templates/edit_book.html:12
+#: bookwyrm/templates/edit_author.html:15 bookwyrm/templates/edit_book.html:15
msgid "Last edited by:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:28 bookwyrm/templates/edit_book.html:27
+#: bookwyrm/templates/edit_author.html:31 bookwyrm/templates/edit_book.html:30
msgid "Metadata"
msgstr ""
-#: bookwyrm/templates/edit_author.html:29 bookwyrm/templates/lists/form.html:8
+#: bookwyrm/templates/edit_author.html:32 bookwyrm/templates/lists/form.html:8
#: bookwyrm/templates/user/create_shelf_form.html:13
#: bookwyrm/templates/user/edit_shelf_form.html:14
msgid "Name:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:34
+#: bookwyrm/templates/edit_author.html:37
msgid "Bio:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:39
+#: bookwyrm/templates/edit_author.html:42
msgid "Wikipedia link:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:44
+#: bookwyrm/templates/edit_author.html:47
msgid "Birth date:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:49
+#: bookwyrm/templates/edit_author.html:52
msgid "Death date:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:55
+#: bookwyrm/templates/edit_author.html:58
msgid "Author Identifiers"
msgstr ""
-#: bookwyrm/templates/edit_author.html:56 bookwyrm/templates/edit_book.html:100
+#: bookwyrm/templates/edit_author.html:59 bookwyrm/templates/edit_book.html:103
msgid "Openlibrary key:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:61
+#: bookwyrm/templates/edit_author.html:64
msgid "Librarything key:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:66
+#: bookwyrm/templates/edit_author.html:69
msgid "Goodreads key:"
msgstr ""
-#: bookwyrm/templates/edit_author.html:76 bookwyrm/templates/edit_book.html:118
-msgid "Cancel"
-msgstr ""
-
-#: bookwyrm/templates/edit_book.html:28
-#: bookwyrm/templates/snippets/create_status_form.html:10
+#: bookwyrm/templates/edit_book.html:31
msgid "Title:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:32
+#: bookwyrm/templates/edit_book.html:35
msgid "Subtitle:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:40
+#: bookwyrm/templates/edit_book.html:43
msgid "Series:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:44
+#: bookwyrm/templates/edit_book.html:47
msgid "Series number:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:48
+#: bookwyrm/templates/edit_book.html:51
msgid "First published date:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:52
+#: bookwyrm/templates/edit_book.html:55
msgid "Published date:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:65
+#: bookwyrm/templates/edit_book.html:68
#: bookwyrm/templates/snippets/shelf.html:9
msgid "Cover"
msgstr ""
-#: bookwyrm/templates/edit_book.html:75
+#: bookwyrm/templates/edit_book.html:78
msgid "Physical Properties"
msgstr ""
-#: bookwyrm/templates/edit_book.html:76
+#: bookwyrm/templates/edit_book.html:79
msgid "Format:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:84
+#: bookwyrm/templates/edit_book.html:87
msgid "Pages:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:91
+#: bookwyrm/templates/edit_book.html:94
msgid "Book Identifiers"
msgstr ""
-#: bookwyrm/templates/edit_book.html:92
+#: bookwyrm/templates/edit_book.html:95
msgid "ISBN 13:"
msgstr ""
-#: bookwyrm/templates/edit_book.html:96
+#: bookwyrm/templates/edit_book.html:99
msgid "ISBN 10:"
msgstr ""
-#: bookwyrm/templates/editions.html:6
+#: bookwyrm/templates/editions.html:5
+#, python-format
+msgid "Editions of %(book_title)s"
+msgstr ""
+
+#: bookwyrm/templates/editions.html:9
#, python-format
msgid "Editions of \"%(work_title)s\" "
msgstr ""
-#: bookwyrm/templates/error.html:6
+#: bookwyrm/templates/error.html:4
+msgid "Oops!"
+msgstr ""
+
+#: bookwyrm/templates/error.html:8
msgid "Server Error"
msgstr ""
-#: bookwyrm/templates/error.html:7
+#: bookwyrm/templates/error.html:9
msgid "Something went wrong! Sorry about that."
msgstr ""
-#: bookwyrm/templates/feed/direct_messages.html:7
+#: bookwyrm/templates/feed/direct_messages.html:8
+#, python-format
+msgid "Direct Messages with %(username)s "
+msgstr ""
+
+#: bookwyrm/templates/feed/direct_messages.html:10
+#: bookwyrm/templates/layout.html:87
+msgid "Direct Messages"
+msgstr ""
+
+#: bookwyrm/templates/feed/direct_messages.html:13
msgid "All messages"
msgstr ""
-#: bookwyrm/templates/feed/direct_messages.html:16
+#: bookwyrm/templates/feed/direct_messages.html:22
msgid "You have no messages right now."
msgstr ""
@@ -292,15 +442,15 @@ msgstr ""
msgid "%(tab_title)s Timeline"
msgstr ""
-#: bookwyrm/templates/feed/feed.html:10
+#: bookwyrm/templates/feed/feed.html:10 bookwyrm/views/feed.py:33
msgid "Home"
msgstr ""
-#: bookwyrm/templates/feed/feed.html:13
+#: bookwyrm/templates/feed/feed.html:13 bookwyrm/views/feed.py:37
msgid "Local"
msgstr ""
-#: bookwyrm/templates/feed/feed.html:16
+#: bookwyrm/templates/feed/feed.html:16 bookwyrm/views/feed.py:41
msgid "Federated"
msgstr ""
@@ -309,25 +459,44 @@ msgid "Announcements"
msgstr ""
#: bookwyrm/templates/feed/feed.html:32
-msgid ""
-"There aren't any activities right now! Try following a user to get started"
+msgid "There aren't any activities right now! Try following a user to get started"
msgstr ""
-#: bookwyrm/templates/feed/feed_layout.html:9
-msgid "Your books"
+#: bookwyrm/templates/feed/feed_layout.html:5
+msgid "Updates"
msgstr ""
#: bookwyrm/templates/feed/feed_layout.html:11
-msgid ""
-"There are no books here right now! Try searching for a book to get started"
+msgid "Your books"
msgstr ""
-#: bookwyrm/templates/feed/feed_layout.html:68
+#: bookwyrm/templates/feed/feed_layout.html:13
+msgid "There are no books here right now! Try searching for a book to get started"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed_layout.html:23
+#: bookwyrm/templates/user/shelf.html:24
+msgid "To Read"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed_layout.html:24
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Currently Reading"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed_layout.html:25
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:11
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Read"
+msgstr ""
+
+#: bookwyrm/templates/feed/feed_layout.html:76 bookwyrm/templates/goal.html:26
+#: bookwyrm/templates/snippets/goal_card.html:6
#, python-format
msgid "%(year)s Reading Goal"
msgstr ""
-#: bookwyrm/templates/feed/status.html:7
+#: bookwyrm/templates/feed/status.html:8
msgid "Back"
msgstr ""
@@ -336,118 +505,165 @@ msgstr ""
msgid "%(year)s Reading Progress"
msgstr ""
-#: bookwyrm/templates/goal.html:29
-#: bookwyrm/templates/snippets/goal_card.html:13
-#, python-format
-msgid ""
-"Set a goal for how many books you'll finish reading in %(year)s, and track "
-"your progress throughout the year."
+#: bookwyrm/templates/goal.html:11
+msgid "Edit Goal"
msgstr ""
-#: bookwyrm/templates/goal.html:38
+#: bookwyrm/templates/goal.html:30
+#: bookwyrm/templates/snippets/goal_card.html:13
+#, python-format
+msgid "Set a goal for how many books you'll finish reading in %(year)s, and track your progress throughout the year."
+msgstr ""
+
+#: bookwyrm/templates/goal.html:39
#, python-format
msgid "%(name)s hasn't set a reading goal for %(year)s."
msgstr ""
-#: bookwyrm/templates/import.html:6
+#: bookwyrm/templates/goal.html:51
+#, python-format
+msgid "Your %(year)s Books"
+msgstr ""
+
+#: bookwyrm/templates/goal.html:53
+#, python-format
+msgid "%(username)s's %(year)s Books"
+msgstr ""
+
+#: bookwyrm/templates/import.html:5 bookwyrm/templates/import.html:9
+#: bookwyrm/templates/layout.html:102
msgid "Import Books"
msgstr ""
-#: bookwyrm/templates/import.html:11
+#: bookwyrm/templates/import.html:14
msgid "Data source"
msgstr ""
-#: bookwyrm/templates/import.html:29
+#: bookwyrm/templates/import.html:32
msgid "Include reviews"
msgstr ""
-#: bookwyrm/templates/import.html:34
+#: bookwyrm/templates/import.html:37
msgid "Privacy setting for imported reviews:"
msgstr ""
-#: bookwyrm/templates/import.html:38
+#: bookwyrm/templates/import.html:41
msgid "Import"
msgstr ""
-#: bookwyrm/templates/import.html:43
+#: bookwyrm/templates/import.html:46
msgid "Recent Imports"
msgstr ""
-#: bookwyrm/templates/import.html:45
+#: bookwyrm/templates/import.html:48
msgid "No recent imports"
msgstr ""
-#: bookwyrm/templates/import_status.html:7
+#: bookwyrm/templates/import_status.html:6
+#: bookwyrm/templates/import_status.html:10
msgid "Import Status"
msgstr ""
-#: bookwyrm/templates/import_status.html:10
+#: bookwyrm/templates/import_status.html:13
msgid "Import started:"
msgstr ""
-#: bookwyrm/templates/import_status.html:14
+#: bookwyrm/templates/import_status.html:17
msgid "Import completed:"
msgstr ""
-#: bookwyrm/templates/import_status.html:17
+#: bookwyrm/templates/import_status.html:20
msgid "TASK FAILED"
msgstr ""
-#: bookwyrm/templates/import_status.html:23
+#: bookwyrm/templates/import_status.html:26
msgid "Import still in progress."
msgstr ""
-#: bookwyrm/templates/import_status.html:25
+#: bookwyrm/templates/import_status.html:28
msgid "(Hit reload to update!)"
msgstr ""
-#: bookwyrm/templates/import_status.html:32
+#: bookwyrm/templates/import_status.html:35
msgid "Failed to load"
msgstr ""
-#: bookwyrm/templates/import_status.html:56
+#: bookwyrm/templates/import_status.html:44
+#, python-format
+msgid "Jump to the bottom of the list to select the %(failed_count)s items which failed to import."
+msgstr ""
+
+#: bookwyrm/templates/import_status.html:79
msgid "Select all"
msgstr ""
-#: bookwyrm/templates/import_status.html:59
+#: bookwyrm/templates/import_status.html:82
msgid "Retry items"
msgstr ""
-#: bookwyrm/templates/import_status.html:81
+#: bookwyrm/templates/import_status.html:108
msgid "Successfully imported"
msgstr ""
-#: bookwyrm/templates/import_status.html:85
+#: bookwyrm/templates/import_status.html:112
#: bookwyrm/templates/lists/curate.html:14
msgid "Book"
msgstr ""
-#: bookwyrm/templates/import_status.html:88
+#: bookwyrm/templates/import_status.html:115
+#: bookwyrm/templates/snippets/create_status_form.html:10
#: bookwyrm/templates/snippets/shelf.html:10
msgid "Title"
msgstr ""
-#: bookwyrm/templates/import_status.html:91
+#: bookwyrm/templates/import_status.html:118
#: bookwyrm/templates/snippets/shelf.html:11
msgid "Author"
msgstr ""
-#: bookwyrm/templates/import_status.html:114
+#: bookwyrm/templates/import_status.html:141
msgid "Imported"
msgstr ""
-#: bookwyrm/templates/invite.html:9 bookwyrm/templates/login.html:41
+#: bookwyrm/templates/invite.html:4 bookwyrm/templates/invite.html:12
+#: bookwyrm/templates/login.html:43
msgid "Create an Account"
msgstr ""
-#: bookwyrm/templates/invite.html:18
+#: bookwyrm/templates/invite.html:21
msgid "Permission Denied"
msgstr ""
-#: bookwyrm/templates/invite.html:19
+#: bookwyrm/templates/invite.html:22
msgid "Sorry! This invite code is no longer valid."
msgstr ""
+#: bookwyrm/templates/isbn_search_results.html:4
+#: bookwyrm/templates/search_results.html:4
+msgid "Search Results"
+msgstr ""
+
+#: bookwyrm/templates/isbn_search_results.html:9
+#: bookwyrm/templates/search_results.html:9
+#, python-format
+msgid "Search Results for \"%(query)s\""
+msgstr ""
+
+#: bookwyrm/templates/isbn_search_results.html:14
+#: bookwyrm/templates/search_results.html:14
+msgid "Matching Books"
+msgstr ""
+
+#: bookwyrm/templates/isbn_search_results.html:17
+#: bookwyrm/templates/search_results.html:17
+#, python-format
+msgid "No books found for \"%(query)s\""
+msgstr ""
+
+#: bookwyrm/templates/layout.html:33
+msgid "Search for a book or user"
+msgstr ""
+
#: bookwyrm/templates/layout.html:37 bookwyrm/templates/layout.html:38
#: bookwyrm/templates/lists/list.html:62
msgid "Search"
@@ -465,34 +681,90 @@ msgstr ""
msgid "Feed"
msgstr ""
-#: bookwyrm/templates/layout.html:125 bookwyrm/templates/layout.html:126
-#: bookwyrm/templates/notifications.html:7
+#: bookwyrm/templates/layout.html:92
+#: bookwyrm/templates/preferences/preferences_layout.html:14
+msgid "Profile"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:97
+msgid "Settings"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:111
+#: bookwyrm/templates/settings/admin_layout.html:19
+#: bookwyrm/templates/settings/manage_invites.html:3
+msgid "Invites"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:118
+msgid "Site Configuration"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:125
+msgid "Log out"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:133 bookwyrm/templates/layout.html:134
+#: bookwyrm/templates/notifications.html:6
+#: bookwyrm/templates/notifications.html:10
msgid "Notifications"
msgstr ""
-#: bookwyrm/templates/layout.html:143 bookwyrm/templates/layout.html:147
-#: bookwyrm/templates/login.html:15
+#: bookwyrm/templates/layout.html:151 bookwyrm/templates/layout.html:155
+#: bookwyrm/templates/login.html:17
#: bookwyrm/templates/snippets/register_form.html:4
msgid "Username:"
msgstr ""
-#: bookwyrm/templates/layout.html:152 bookwyrm/templates/login.html:8
-#: bookwyrm/templates/login.html:31
+#: bookwyrm/templates/layout.html:156
+msgid "password"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:157 bookwyrm/templates/login.html:36
+msgid "Forgot your password?"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:160 bookwyrm/templates/login.html:10
+#: bookwyrm/templates/login.html:33
msgid "Log in"
msgstr ""
-#: bookwyrm/templates/layout.html:183
+#: bookwyrm/templates/layout.html:168
+msgid "Join"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:191
msgid "About this server"
msgstr ""
-#: bookwyrm/templates/layout.html:187
+#: bookwyrm/templates/layout.html:195
msgid "Contact site admin"
msgstr ""
+#: bookwyrm/templates/layout.html:202
+#, python-format
+msgid "Support %(site_name)s on %(support_title)s "
+msgstr ""
+
+#: bookwyrm/templates/layout.html:206
+msgid "BookWyrm is open source software. You can contribute or report issues on GitHub ."
+msgstr ""
+
#: bookwyrm/templates/lists/create_form.html:5
+#: bookwyrm/templates/lists/lists.html:17
msgid "Create List"
msgstr ""
+#: bookwyrm/templates/lists/created_text.html:5
+#, python-format
+msgid "Created and curated by %(username)s "
+msgstr ""
+
+#: bookwyrm/templates/lists/created_text.html:7
+#, python-format
+msgid "Created by %(username)s "
+msgstr ""
+
#: bookwyrm/templates/lists/curate.html:6
msgid "Pending Books"
msgstr ""
@@ -518,6 +790,7 @@ msgid "Discard"
msgstr ""
#: bookwyrm/templates/lists/edit_form.html:5
+#: bookwyrm/templates/lists/list_layout.html:18
msgid "Edit List"
msgstr ""
@@ -554,7 +827,8 @@ msgid "This list is currently empty"
msgstr ""
#: bookwyrm/templates/lists/list.html:35
-msgid "Added by"
+#, python-format
+msgid "Added by %(username)s "
msgstr ""
#: bookwyrm/templates/lists/list.html:41
@@ -594,166 +868,224 @@ msgstr ""
msgid "Suggest"
msgstr ""
-#: bookwyrm/templates/lists/list_items.html:19
-#: bookwyrm/templates/lists/list_layout.html:9
-msgid "Created and curated by"
-msgstr ""
-
-#: bookwyrm/templates/lists/list_items.html:19
-#: bookwyrm/templates/lists/list_layout.html:9
-msgid "Created by"
-msgstr ""
-
-#: bookwyrm/templates/lists/lists.html:11
+#: bookwyrm/templates/lists/lists.html:14
msgid "Your lists"
msgstr ""
-#: bookwyrm/templates/lists/lists.html:36
+#: bookwyrm/templates/lists/lists.html:32
+#, python-format
+msgid "See all %(size)s lists"
+msgstr ""
+
+#: bookwyrm/templates/lists/lists.html:40
msgid "Recent Lists"
msgstr ""
-#: bookwyrm/templates/login.html:21 bookwyrm/templates/password_reset.html:15
+#: bookwyrm/templates/login.html:4
+msgid "Login"
+msgstr ""
+
+#: bookwyrm/templates/login.html:23 bookwyrm/templates/password_reset.html:17
#: bookwyrm/templates/snippets/register_form.html:22
msgid "Password:"
msgstr ""
-#: bookwyrm/templates/login.html:34
-msgid "Forgot your password?"
-msgstr ""
-
-#: bookwyrm/templates/login.html:47
+#: bookwyrm/templates/login.html:49
msgid "Contact an administrator to get an invite"
msgstr ""
-#: bookwyrm/templates/login.html:57
+#: bookwyrm/templates/login.html:59
msgid "More about this site"
msgstr ""
-#: bookwyrm/templates/notfound.html:6
+#: bookwyrm/templates/notfound.html:4 bookwyrm/templates/notfound.html:8
msgid "Not Found"
msgstr ""
-#: bookwyrm/templates/notfound.html:7
-msgid "The page your requested doesn't seem to exist!"
+#: bookwyrm/templates/notfound.html:9
+msgid "The page you requested doesn't seem to exist!"
msgstr ""
-#: bookwyrm/templates/notifications.html:11
+#: bookwyrm/templates/notifications.html:14
msgid "Delete notifications"
msgstr ""
-#: bookwyrm/templates/notifications.html:45
-#, python-format
-msgid "favorited your %(preview_name)s "
-msgstr ""
-
-#: bookwyrm/templates/notifications.html:48
-#, python-format
-msgid "mentioned you in a %(preview_name)s "
-msgstr ""
-
#: bookwyrm/templates/notifications.html:51
#, python-format
-msgid ""
-"replied to your "
-"%(preview_name)s "
+msgid "favorited your review of %(book_title)s "
msgstr ""
-#: bookwyrm/templates/notifications.html:54
-msgid "followed you"
+#: bookwyrm/templates/notifications.html:53
+#, python-format
+msgid "favorited your comment on %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:55
+#, python-format
+msgid "favorited your quote from %(book_title)s "
msgstr ""
#: bookwyrm/templates/notifications.html:57
-msgid "sent you a follow request"
+#, python-format
+msgid "favorited your status "
msgstr ""
#: bookwyrm/templates/notifications.html:62
#, python-format
-msgid "boosted your %(preview_name)s "
+msgid "mentioned you in a review of %(book_title)s "
msgstr ""
#: bookwyrm/templates/notifications.html:64
-msgid "added"
+#, python-format
+msgid "mentioned you in a comment on %(book_title)s "
msgstr ""
-#: bookwyrm/templates/notifications.html:64
-msgid "suggested adding"
+#: bookwyrm/templates/notifications.html:66
+#, python-format
+msgid "mentioned you in a quote from %(book_title)s "
msgstr ""
-#: bookwyrm/templates/notifications.html:67
+#: bookwyrm/templates/notifications.html:68
+#, python-format
+msgid "mentioned you in a status "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:73
+#, python-format
+msgid "replied to your review of %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:75
+#, python-format
+msgid "replied to your comment on %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:77
+#, python-format
+msgid "replied to your quote from %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:79
+#, python-format
+msgid "replied to your status "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:83
+msgid "followed you"
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:86
+msgid "sent you a follow request"
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:92
+#, python-format
+msgid "boosted your review of %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:94
+#, python-format
+msgid "boosted your comment on%(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:96
+#, python-format
+msgid "boosted your quote from %(book_title)s "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:98
+#, python-format
+msgid "boosted your status "
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:102
+#, python-format
+msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:104
+#, python-format
+msgid " suggested adding %(book_title)s to your list \"%(list_name)s \""
+msgstr ""
+
+#: bookwyrm/templates/notifications.html:108
#, python-format
msgid " your import completed."
msgstr ""
-#: bookwyrm/templates/notifications.html:99
+#: bookwyrm/templates/notifications.html:142
msgid "You're all caught up!"
msgstr ""
-#: bookwyrm/templates/password_reset.html:8
-#: bookwyrm/templates/password_reset_request.html:8
+#: bookwyrm/templates/password_reset.html:4
+#: bookwyrm/templates/password_reset.html:10
+#: bookwyrm/templates/password_reset_request.html:4
+#: bookwyrm/templates/password_reset_request.html:10
msgid "Reset Password"
msgstr ""
-#: bookwyrm/templates/password_reset.html:21
-#: bookwyrm/templates/preferences/change_password.html:15
+#: bookwyrm/templates/password_reset.html:23
+#: bookwyrm/templates/preferences/change_password.html:18
msgid "Confirm password:"
msgstr ""
-#: bookwyrm/templates/password_reset.html:28
+#: bookwyrm/templates/password_reset.html:30
msgid "Confirm"
msgstr ""
-#: bookwyrm/templates/password_reset_request.html:10
+#: bookwyrm/templates/password_reset_request.html:12
msgid "A link to reset your password will be sent to your email address"
msgstr ""
-#: bookwyrm/templates/password_reset_request.html:14
-#: bookwyrm/templates/preferences/edit_user.html:35
+#: bookwyrm/templates/password_reset_request.html:16
+#: bookwyrm/templates/preferences/edit_user.html:38
#: bookwyrm/templates/snippets/register_form.html:13
msgid "Email address:"
msgstr ""
-#: bookwyrm/templates/password_reset_request.html:21
+#: bookwyrm/templates/password_reset_request.html:23
msgid "Reset password"
msgstr ""
-#: bookwyrm/templates/preferences/blocks.html:5
+#: bookwyrm/templates/preferences/blocks.html:4
+#: bookwyrm/templates/preferences/blocks.html:7
+#: bookwyrm/templates/preferences/preferences_layout.html:23
msgid "Blocked Users"
msgstr ""
-#: bookwyrm/templates/preferences/blocks.html:10
+#: bookwyrm/templates/preferences/blocks.html:12
msgid "No users currently blocked."
msgstr ""
#: bookwyrm/templates/preferences/change_password.html:4
+#: bookwyrm/templates/preferences/change_password.html:7
+#: bookwyrm/templates/preferences/change_password.html:21
+#: bookwyrm/templates/preferences/preferences_layout.html:17
msgid "Change Password"
msgstr ""
-#: bookwyrm/templates/preferences/change_password.html:11
+#: bookwyrm/templates/preferences/change_password.html:14
msgid "New password:"
msgstr ""
-#: bookwyrm/templates/preferences/change_password.html:18
-#: bookwyrm/templates/preferences/preferences_layout.html:17
-msgid "Change password"
-msgstr ""
-
#: bookwyrm/templates/preferences/edit_user.html:4
+#: bookwyrm/templates/preferences/edit_user.html:7
msgid "Edit Profile"
msgstr ""
-#: bookwyrm/templates/preferences/edit_user.html:14
+#: bookwyrm/templates/preferences/edit_user.html:17
msgid "Avatar:"
msgstr ""
-#: bookwyrm/templates/preferences/edit_user.html:21
+#: bookwyrm/templates/preferences/edit_user.html:24
msgid "Display name:"
msgstr ""
-#: bookwyrm/templates/preferences/edit_user.html:28
+#: bookwyrm/templates/preferences/edit_user.html:31
msgid "Summary:"
msgstr ""
-#: bookwyrm/templates/preferences/edit_user.html:43
+#: bookwyrm/templates/preferences/edit_user.html:46
msgid "Manually approve followers:"
msgstr ""
@@ -761,106 +1093,92 @@ msgstr ""
msgid "Account"
msgstr ""
-#: bookwyrm/templates/preferences/preferences_layout.html:14
-msgid "Profile"
-msgstr ""
-
#: bookwyrm/templates/preferences/preferences_layout.html:20
msgid "Relationships"
msgstr ""
-#: bookwyrm/templates/preferences/preferences_layout.html:23
-msgid "Blocked users"
-msgstr ""
-
-#: bookwyrm/templates/search_results.html:6
-#, python-format
-msgid "Search Results for \"%(query)s\""
-msgstr ""
-
-#: bookwyrm/templates/search_results.html:11
-msgid "Matching Books"
-msgstr ""
-
-#: bookwyrm/templates/search_results.html:14
-#, python-format
-msgid "No books found for \"%(query)s\""
-msgstr ""
-
-#: bookwyrm/templates/search_results.html:30
+#: bookwyrm/templates/search_results.html:33
msgid "Didn't find what you were looking for?"
msgstr ""
-#: bookwyrm/templates/search_results.html:53
+#: bookwyrm/templates/search_results.html:35
+msgid "Show results from other catalogues"
+msgstr ""
+
+#: bookwyrm/templates/search_results.html:57
msgid "Import book"
msgstr ""
-#: bookwyrm/templates/search_results.html:70
+#: bookwyrm/templates/search_results.html:67
+msgid "Hide results from other catalogues"
+msgstr ""
+
+#: bookwyrm/templates/search_results.html:75
msgid "Matching Users"
msgstr ""
-#: bookwyrm/templates/search_results.html:72
+#: bookwyrm/templates/search_results.html:77
#, python-format
msgid "No users found for \"%(query)s\""
msgstr ""
-#: bookwyrm/templates/search_results.html:87
+#: bookwyrm/templates/search_results.html:94
#, python-format
msgid "No lists found for \"%(query)s\""
msgstr ""
-#: bookwyrm/templates/settings/admin_layout.html:12
+#: bookwyrm/templates/settings/admin_layout.html:4
+msgid "Administration"
+msgstr ""
+
+#: bookwyrm/templates/settings/admin_layout.html:15
msgid "Manage Users"
msgstr ""
-#: bookwyrm/templates/settings/admin_layout.html:16
-#: bookwyrm/templates/settings/manage_invites.html:3
-msgid "Invites"
-msgstr ""
-
-#: bookwyrm/templates/settings/admin_layout.html:20
-#: bookwyrm/templates/settings/federation.html:3
+#: bookwyrm/templates/settings/admin_layout.html:23
+#: bookwyrm/templates/settings/federation.html:4
msgid "Federated Servers"
msgstr ""
-#: bookwyrm/templates/settings/admin_layout.html:25
+#: bookwyrm/templates/settings/admin_layout.html:28
msgid "Instance Settings"
msgstr ""
-#: bookwyrm/templates/settings/admin_layout.html:29
-#: bookwyrm/templates/settings/site.html:3
-msgid "Site Configuration"
-msgstr ""
-
#: bookwyrm/templates/settings/admin_layout.html:32
-#: bookwyrm/templates/settings/site.html:10
-msgid "Instance Info"
-msgstr ""
-
-#: bookwyrm/templates/settings/admin_layout.html:33
-#: bookwyrm/templates/settings/site.html:36
-msgid "Images"
-msgstr ""
-
-#: bookwyrm/templates/settings/admin_layout.html:34
-#: bookwyrm/templates/settings/site.html:56
-msgid "Footer Content"
+#: bookwyrm/templates/settings/site.html:4
+#: bookwyrm/templates/settings/site.html:6
+msgid "Site Settings"
msgstr ""
#: bookwyrm/templates/settings/admin_layout.html:35
-#: bookwyrm/templates/settings/site.html:74
+#: bookwyrm/templates/settings/site.html:13
+msgid "Instance Info"
+msgstr ""
+
+#: bookwyrm/templates/settings/admin_layout.html:36
+#: bookwyrm/templates/settings/site.html:39
+msgid "Images"
+msgstr ""
+
+#: bookwyrm/templates/settings/admin_layout.html:37
+#: bookwyrm/templates/settings/site.html:59
+msgid "Footer Content"
+msgstr ""
+
+#: bookwyrm/templates/settings/admin_layout.html:38
+#: bookwyrm/templates/settings/site.html:77
msgid "Registration"
msgstr ""
-#: bookwyrm/templates/settings/federation.html:9
+#: bookwyrm/templates/settings/federation.html:10
msgid "Server name"
msgstr ""
-#: bookwyrm/templates/settings/federation.html:10
+#: bookwyrm/templates/settings/federation.html:11
msgid "Software"
msgstr ""
-#: bookwyrm/templates/settings/federation.html:11
+#: bookwyrm/templates/settings/federation.html:12
msgid "Status"
msgstr ""
@@ -900,55 +1218,55 @@ msgstr ""
msgid "No active invites"
msgstr ""
-#: bookwyrm/templates/settings/site.html:12
+#: bookwyrm/templates/settings/site.html:15
msgid "Instance Name:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:16
+#: bookwyrm/templates/settings/site.html:19
msgid "Tagline:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:20
+#: bookwyrm/templates/settings/site.html:23
msgid "Instance description:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:24
+#: bookwyrm/templates/settings/site.html:27
msgid "Code of conduct:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:28
+#: bookwyrm/templates/settings/site.html:31
msgid "Privacy Policy:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:39
+#: bookwyrm/templates/settings/site.html:42
msgid "Logo:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:43
+#: bookwyrm/templates/settings/site.html:46
msgid "Logo small:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:47
+#: bookwyrm/templates/settings/site.html:50
msgid "Favicon:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:58
+#: bookwyrm/templates/settings/site.html:61
msgid "Support link:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:62
+#: bookwyrm/templates/settings/site.html:65
msgid "Support title:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:66
+#: bookwyrm/templates/settings/site.html:69
msgid "Admin email:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:76
+#: bookwyrm/templates/settings/site.html:79
msgid "Allow registration:"
msgstr ""
-#: bookwyrm/templates/settings/site.html:80
+#: bookwyrm/templates/settings/site.html:83
msgid "Registration closed text:"
msgstr ""
@@ -960,10 +1278,15 @@ msgstr ""
msgid "Un-block"
msgstr ""
+#: bookwyrm/templates/snippets/book_titleby.html:3
+#, python-format
+msgid "%(title)s by "
+msgstr ""
+
#: bookwyrm/templates/snippets/boost_button.html:8
#: bookwyrm/templates/snippets/boost_button.html:9
-#: bookwyrm/templates/snippets/status/status_body.html:40
#: bookwyrm/templates/snippets/status/status_body.html:41
+#: bookwyrm/templates/snippets/status/status_body.html:42
msgid "Boost status"
msgstr ""
@@ -980,15 +1303,16 @@ msgstr ""
msgid "Spoilers ahead!"
msgstr ""
-#: bookwyrm/templates/snippets/create_status.html:8
+#: bookwyrm/templates/snippets/create_status.html:9
msgid "Review"
msgstr ""
-#: bookwyrm/templates/snippets/create_status.html:11
+#: bookwyrm/templates/snippets/create_status.html:12
+#: bookwyrm/templates/snippets/create_status_form.html:44
msgid "Comment"
msgstr ""
-#: bookwyrm/templates/snippets/create_status.html:14
+#: bookwyrm/templates/snippets/create_status.html:15
msgid "Quote"
msgstr ""
@@ -1003,18 +1327,18 @@ msgstr ""
msgid "No rating"
msgstr ""
-#: bookwyrm/templates/snippets/create_status_form.html:44
-msgid "Comment:"
+#: bookwyrm/templates/snippets/create_status_form.html:54
+msgid "Include spoiler alert"
msgstr ""
-#: bookwyrm/templates/snippets/create_status_form.html:59
+#: bookwyrm/templates/snippets/create_status_form.html:60
#: bookwyrm/templates/snippets/privacy-icons.html:15
#: bookwyrm/templates/snippets/privacy-icons.html:16
#: bookwyrm/templates/snippets/privacy_select.html:19
msgid "Private"
msgstr ""
-#: bookwyrm/templates/snippets/create_status_form.html:66
+#: bookwyrm/templates/snippets/create_status_form.html:67
msgid "Post"
msgstr ""
@@ -1024,9 +1348,7 @@ msgstr ""
#: bookwyrm/templates/snippets/delete_readthrough_modal.html:7
#, python-format
-msgid ""
-"You are deleting this readthrough and its %(count)s associated progress "
-"updates."
+msgid "You are deleting this readthrough and its %(count)s associated progress updates."
msgstr ""
#: bookwyrm/templates/snippets/delete_readthrough_modal.html:15
@@ -1036,8 +1358,8 @@ msgstr ""
#: bookwyrm/templates/snippets/fav_button.html:7
#: bookwyrm/templates/snippets/fav_button.html:8
-#: bookwyrm/templates/snippets/status/status_body.html:44
#: bookwyrm/templates/snippets/status/status_body.html:45
+#: bookwyrm/templates/snippets/status/status_body.html:46
msgid "Like status"
msgstr ""
@@ -1066,10 +1388,12 @@ msgstr ""
msgid "Accept"
msgstr ""
-#: bookwyrm/templates/snippets/goal_card.html:6
+#: bookwyrm/templates/snippets/generated_status/goal.html:1
#, python-format
-msgid "%(year)s reading goal"
-msgstr ""
+msgid "set a goal to read %(counter)s book in %(year)s"
+msgid_plural "set a goal to read %(counter)s books in %(year)s"
+msgstr[0] ""
+msgstr[1] ""
#: bookwyrm/templates/snippets/goal_card.html:21
msgid "Dismiss message"
@@ -1077,9 +1401,7 @@ msgstr ""
#: bookwyrm/templates/snippets/goal_card.html:22
#, python-format
-msgid ""
-"You can set or change your reading goal any time from your profile page "
+msgid "You can set or change your reading goal any time from your profile page "
msgstr ""
#: bookwyrm/templates/snippets/goal_form.html:9
@@ -1114,6 +1436,16 @@ msgstr ""
msgid "%(percent)s%% complete!"
msgstr ""
+#: bookwyrm/templates/snippets/goal_progress.html:10
+#, python-format
+msgid "You've read %(read_count)s of %(goal_count)s books ."
+msgstr ""
+
+#: bookwyrm/templates/snippets/goal_progress.html:12
+#, python-format
+msgid "%(username)s has read %(read_count)s of %(goal_count)s books ."
+msgstr ""
+
#: bookwyrm/templates/snippets/pagination.html:7
msgid "Previous"
msgstr ""
@@ -1143,7 +1475,7 @@ msgid "Post privacy"
msgstr ""
#: bookwyrm/templates/snippets/privacy_select.html:16
-#: bookwyrm/templates/user/followers.html:17
+#: bookwyrm/templates/user/followers.html:13
msgid "Followers"
msgstr ""
@@ -1182,18 +1514,27 @@ msgstr ""
msgid "finished"
msgstr ""
-#: bookwyrm/templates/snippets/readthrough.html:29
+#: bookwyrm/templates/snippets/readthrough.html:14
+msgid "Show all updates"
+msgstr ""
+
+#: bookwyrm/templates/snippets/readthrough.html:30
msgid "Delete this progress update"
msgstr ""
-#: bookwyrm/templates/snippets/readthrough.html:39
+#: bookwyrm/templates/snippets/readthrough.html:40
msgid "started"
msgstr ""
-#: bookwyrm/templates/snippets/readthrough.html:57
+#: bookwyrm/templates/snippets/readthrough.html:46
+#: bookwyrm/templates/snippets/readthrough.html:60
msgid "Edit read dates"
msgstr ""
+#: bookwyrm/templates/snippets/readthrough.html:50
+msgid "Delete these read dates"
+msgstr ""
+
#: bookwyrm/templates/snippets/readthrough_form.html:7
#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:19
#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:17
@@ -1214,22 +1555,22 @@ msgid "Sign Up"
msgstr ""
#: bookwyrm/templates/snippets/rss_title.html:5
-#: bookwyrm/templates/snippets/status/status_header.html:9
+#: bookwyrm/templates/snippets/status/status_header.html:11
msgid "rated"
msgstr ""
#: bookwyrm/templates/snippets/rss_title.html:7
-#: bookwyrm/templates/snippets/status/status_header.html:11
+#: bookwyrm/templates/snippets/status/status_header.html:13
msgid "reviewed"
msgstr ""
#: bookwyrm/templates/snippets/rss_title.html:9
-#: bookwyrm/templates/snippets/status/status_header.html:13
+#: bookwyrm/templates/snippets/status/status_header.html:15
msgid "commented on"
msgstr ""
#: bookwyrm/templates/snippets/rss_title.html:11
-#: bookwyrm/templates/snippets/status/status_header.html:15
+#: bookwyrm/templates/snippets/status/status_header.html:17
msgid "quoted"
msgstr ""
@@ -1287,8 +1628,17 @@ msgstr ""
msgid "More shelves"
msgstr ""
-#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:10
-msgid "Read"
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:8
+msgid "Start reading"
+msgstr ""
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:13
+msgid "Finish reading"
+msgstr ""
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:16
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
+msgid "Want to read"
msgstr ""
#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:5
@@ -1301,30 +1651,57 @@ msgstr ""
msgid "Want to Read \"%(book_title)s \""
msgstr ""
-#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
-msgid "Want to read"
-msgstr ""
-
-#: bookwyrm/templates/snippets/status/status.html:7
+#: bookwyrm/templates/snippets/status/status.html:9
msgid "boosted"
msgstr ""
-#: bookwyrm/templates/snippets/status/status_body.html:36
+#: bookwyrm/templates/snippets/status/status_body.html:24
#: bookwyrm/templates/snippets/status/status_body.html:37
+#: bookwyrm/templates/snippets/status/status_body.html:38
msgid "Reply"
msgstr ""
-#: bookwyrm/templates/snippets/status/status_content.html:42
+#: bookwyrm/templates/snippets/status/status_content.html:18
+#: bookwyrm/templates/snippets/trimmed_text.html:15
+msgid "Show more"
+msgstr ""
+
+#: bookwyrm/templates/snippets/status/status_content.html:25
+#: bookwyrm/templates/snippets/trimmed_text.html:25
+msgid "Show less"
+msgstr ""
+
+#: bookwyrm/templates/snippets/status/status_content.html:46
msgid "Open image in new window"
msgstr ""
+#: bookwyrm/templates/snippets/status/status_header.html:22
+#, python-format
+msgid "replied to %(username)s's review "
+msgstr ""
+
+#: bookwyrm/templates/snippets/status/status_header.html:24
+#, python-format
+msgid "replied to %(username)s's comment "
+msgstr ""
+
+#: bookwyrm/templates/snippets/status/status_header.html:26
+#, python-format
+msgid "replied to %(username)s's quote "
+msgstr ""
+
+#: bookwyrm/templates/snippets/status/status_header.html:28
+#, python-format
+msgid "replied to %(username)s's status "
+msgstr ""
+
#: bookwyrm/templates/snippets/status/status_options.html:7
#: bookwyrm/templates/snippets/user_options.html:7
msgid "More options"
msgstr ""
#: bookwyrm/templates/snippets/status/status_options.html:17
-msgid "Delete post"
+msgid "Delete status"
msgstr ""
#: bookwyrm/templates/snippets/status/status_options.html:23
@@ -1340,17 +1717,14 @@ msgstr ""
msgid "Remove tag"
msgstr ""
-#: bookwyrm/templates/tag.html:7
+#: bookwyrm/templates/tag.html:9
#, python-format
msgid "Books tagged \"%(tag.name)s\""
msgstr ""
#: bookwyrm/templates/user/create_shelf_form.html:5
-msgid "Create New Shelf"
-msgstr ""
-
#: bookwyrm/templates/user/create_shelf_form.html:22
-msgid "Create shelf"
+msgid "Create Shelf"
msgstr ""
#: bookwyrm/templates/user/edit_shelf_form.html:5
@@ -1361,73 +1735,100 @@ msgstr ""
msgid "Update shelf"
msgstr ""
-#: bookwyrm/templates/user/followers.html:30
+#: bookwyrm/templates/user/followers.html:7
+#: bookwyrm/templates/user/following.html:7 bookwyrm/templates/user/user.html:9
+msgid "User Profile"
+msgstr ""
+
+#: bookwyrm/templates/user/followers.html:29
#, python-format
msgid "%(username)s has no followers"
msgstr ""
-#: bookwyrm/templates/user/following.html:17
+#: bookwyrm/templates/user/following.html:13
msgid "Following"
msgstr ""
-#: bookwyrm/templates/user/following.html:30
+#: bookwyrm/templates/user/following.html:29
#, python-format
msgid "%(username)s isn't following any users"
msgstr ""
-#: bookwyrm/templates/user/lists.html:28
+#: bookwyrm/templates/user/lists.html:9
+msgid "Your Lists"
+msgstr ""
+
+#: bookwyrm/templates/user/lists.html:11
+#, python-format
+msgid "Lists: %(username)s"
+msgstr ""
+
+#: bookwyrm/templates/user/lists.html:17 bookwyrm/templates/user/lists.html:29
msgid "Create list"
msgstr ""
-#: bookwyrm/templates/user/user.html:7
-msgid "User profile"
+#: bookwyrm/templates/user/shelf.html:9
+msgid "Your Shelves"
msgstr ""
-#: bookwyrm/templates/user/user.html:13
+#: bookwyrm/templates/user/shelf.html:11
+#, python-format
+msgid "%(username)s: Shelves"
+msgstr ""
+
+#: bookwyrm/templates/user/shelf.html:33
+msgid "Create shelf"
+msgstr ""
+
+#: bookwyrm/templates/user/shelf.html:54
+msgid "Edit shelf"
+msgstr ""
+
+#: bookwyrm/templates/user/user.html:15
msgid "Edit profile"
msgstr ""
-#: bookwyrm/templates/user/user.html:24
-#: bookwyrm/templates/user/user_layout.html:66
+#: bookwyrm/templates/user/user.html:26
+#: bookwyrm/templates/user/user_layout.html:68
msgid "Shelves"
msgstr ""
-#: bookwyrm/templates/user/user.html:29
+#: bookwyrm/templates/user/user.html:31
#, python-format
msgid "See all %(size)s"
msgstr ""
-#: bookwyrm/templates/user/user.html:42
+#: bookwyrm/templates/user/user.html:44
#, python-format
msgid "See all %(shelf_count)s shelves"
msgstr ""
-#: bookwyrm/templates/user/user.html:54
+#: bookwyrm/templates/user/user.html:56
#, python-format
msgid "Set a reading goal for %(year)s"
msgstr ""
-#: bookwyrm/templates/user/user.html:60
+#: bookwyrm/templates/user/user.html:62
msgid "User Activity"
msgstr ""
-#: bookwyrm/templates/user/user.html:63
+#: bookwyrm/templates/user/user.html:65
msgid "RSS feed"
msgstr ""
-#: bookwyrm/templates/user/user.html:74
+#: bookwyrm/templates/user/user.html:76
msgid "No activities yet!"
msgstr ""
-#: bookwyrm/templates/user/user_layout.html:30
+#: bookwyrm/templates/user/user_layout.html:32
msgid "Follow Requests"
msgstr ""
-#: bookwyrm/templates/user/user_layout.html:48
+#: bookwyrm/templates/user/user_layout.html:50
msgid "Activity"
msgstr ""
-#: bookwyrm/templates/user/user_layout.html:54
+#: bookwyrm/templates/user/user_layout.html:56
msgid "Reading Goal"
msgstr ""
@@ -1435,3 +1836,15 @@ msgstr ""
#, python-format
msgid "Joined %(date)s"
msgstr ""
+
+#: bookwyrm/templates/user/user_preview.html:15
+#, python-format
+msgid "%(counter)s follower"
+msgid_plural "%(counter)s followers"
+msgstr[0] ""
+msgstr[1] ""
+
+#: bookwyrm/templates/user/user_preview.html:16
+#, python-format
+msgid "%(counter)s following"
+msgstr ""
diff --git a/locale/es/LC_MESSAGES/django.mo b/locale/es/LC_MESSAGES/django.mo
new file mode 100644
index 00000000..0a0af7d1
Binary files /dev/null and b/locale/es/LC_MESSAGES/django.mo differ
diff --git a/locale/es/LC_MESSAGES/django.po b/locale/es/LC_MESSAGES/django.po
new file mode 100644
index 00000000..6c6ee434
--- /dev/null
+++ b/locale/es/LC_MESSAGES/django.po
@@ -0,0 +1,1871 @@
+# SOME DESCRIPTIVE TITLE.
+# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER
+# This file is distributed under the same license as the PACKAGE package.
+# FIRST AUTHOR , YEAR.
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: PACKAGE VERSION\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2021-03-07 23:40+0000\n"
+"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
+"Last-Translator: FULL NAME \n"
+"Language-Team: LANGUAGE \n"
+"Language: \n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=(n != 1);\n"
+
+#: bookwyrm/forms.py:185
+msgid "One Day"
+msgstr "Un día"
+
+#: bookwyrm/forms.py:186
+msgid "One Week"
+msgstr "Una semana"
+
+#: bookwyrm/forms.py:187
+msgid "One Month"
+msgstr "Un mes"
+
+#: bookwyrm/forms.py:188
+msgid "Does Not Expire"
+msgstr "Nunca se vence"
+
+#: bookwyrm/forms.py:190
+#, python-format
+msgid "%(count)d uses"
+msgstr "%(count)d usos"
+
+#: bookwyrm/forms.py:192
+msgid "Unlimited"
+msgstr "Sin límite"
+
+#: bookwyrm/models/fields.py:24
+#, python-format
+msgid "%(value)s is not a valid remote_id"
+msgstr "%(value)s no es un remote_id válido"
+
+#: bookwyrm/models/fields.py:33 bookwyrm/models/fields.py:42
+#, python-format
+msgid "%(value)s is not a valid username"
+msgstr "%(value)s no es un usuario válido"
+
+#: bookwyrm/models/fields.py:164 bookwyrm/templates/layout.html:152
+msgid "username"
+msgstr "nombre de usuario"
+
+#: bookwyrm/models/fields.py:169
+msgid "A user with that username already exists."
+msgstr "Ya existe un usuario con ese nombre."
+
+#: bookwyrm/settings.py:142
+msgid "English"
+msgstr "Inglés"
+
+#: bookwyrm/settings.py:143
+msgid "German"
+msgstr "Aléman"
+
+#: bookwyrm/settings.py:144
+msgid "Spanish"
+msgstr "Español"
+
+#: bookwyrm/settings.py:145
+msgid "French"
+msgstr "Francés"
+
+#: bookwyrm/settings.py:146
+msgid "Simplified Chinese"
+msgstr "Chino simplificado"
+
+#: bookwyrm/templates/author.html:16 bookwyrm/templates/author.html:17
+#: bookwyrm/templates/edit_author.html:5
+msgid "Edit Author"
+msgstr "Editar Autor/Autora"
+
+#: bookwyrm/templates/author.html:32
+msgid "Wikipedia"
+msgstr "Wikipedia"
+
+#: bookwyrm/templates/author.html:37
+#, python-format
+msgid "Books by %(name)s"
+msgstr "Libros de %(name)s"
+
+#: bookwyrm/templates/book.html:21
+#: bookwyrm/templates/discover/large-book.html:12
+#: bookwyrm/templates/discover/small-book.html:9
+msgid "by"
+msgstr "por"
+
+#: bookwyrm/templates/book.html:29 bookwyrm/templates/book.html:30
+#: bookwyrm/templates/edit_book.html:5
+msgid "Edit Book"
+msgstr "Editar Libro"
+
+#: bookwyrm/templates/book.html:45
+msgid "Add cover"
+msgstr "Agregar portada"
+
+#: bookwyrm/templates/book.html:51 bookwyrm/templates/lists/list.html:89
+msgid "Add"
+msgstr "Agregar"
+
+#: bookwyrm/templates/book.html:60
+msgid "ISBN:"
+msgstr "ISBN:"
+
+#: bookwyrm/templates/book.html:67 bookwyrm/templates/edit_book.html:107
+msgid "OCLC Number:"
+msgstr "Número OCLC:"
+
+#: bookwyrm/templates/book.html:74 bookwyrm/templates/edit_book.html:111
+msgid "ASIN:"
+msgstr "ASIN:"
+
+#: bookwyrm/templates/book.html:84
+#, python-format
+msgid "%(format)s, %(pages)s pages"
+msgstr "%(format)s, %(pages)s páginas"
+
+#: bookwyrm/templates/book.html:86
+#, python-format
+msgid "%(pages)s pages"
+msgstr "%(pages)s páginas"
+
+#: bookwyrm/templates/book.html:91
+msgid "View on OpenLibrary"
+msgstr "Ver en OpenLibrary"
+
+#: bookwyrm/templates/book.html:100
+#, python-format
+msgid "(%(review_count)s review)"
+msgid_plural "(%(review_count)s reviews)"
+msgstr[0] "(%(review_count)s reseña)"
+msgstr[1] "(%(review_count)s reseñas)"
+
+#: bookwyrm/templates/book.html:106
+msgid "Add Description"
+msgstr "Agregar descripción"
+
+#: bookwyrm/templates/book.html:113 bookwyrm/templates/edit_book.html:39
+#: bookwyrm/templates/lists/form.html:12
+msgid "Description:"
+msgstr "Descripción:"
+
+#: bookwyrm/templates/book.html:117 bookwyrm/templates/edit_author.html:78
+#: bookwyrm/templates/edit_book.html:120 bookwyrm/templates/lists/form.html:42
+#: bookwyrm/templates/preferences/edit_user.html:50
+#: bookwyrm/templates/settings/site.html:89
+#: bookwyrm/templates/snippets/progress_update.html:21
+#: bookwyrm/templates/snippets/readthrough.html:64
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:42
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:34
+msgid "Save"
+msgstr "Guardar"
+
+#: bookwyrm/templates/book.html:118 bookwyrm/templates/book.html:167
+#: bookwyrm/templates/edit_author.html:79 bookwyrm/templates/edit_book.html:121
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:17
+#: bookwyrm/templates/snippets/goal_form.html:32
+#: bookwyrm/templates/snippets/readthrough.html:65
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:43
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:35
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:28
+msgid "Cancel"
+msgstr "Cancelar"
+
+#: bookwyrm/templates/book.html:127
+#, python-format
+msgid "%(count)s editions "
+msgstr "%(count)s ediciones "
+
+#: bookwyrm/templates/book.html:135
+#, python-format
+msgid "This edition is on your %(shelf_name)s shelf."
+msgstr "Esta edición está en tu %(shelf_name)s estante."
+
+#: bookwyrm/templates/book.html:141
+#, python-format
+msgid "A different edition of this book is on your %(shelf_name)s shelf."
+msgstr "Una edición diferente de este libro está en tu %(shelf_name)s estante."
+
+#: bookwyrm/templates/book.html:150
+msgid "Your reading activity"
+msgstr "Tu actividad de lectura"
+
+#: bookwyrm/templates/book.html:152
+msgid "Add read dates"
+msgstr "Agregar fechas de lectura"
+
+#: bookwyrm/templates/book.html:157
+msgid "You don't have any reading activity for this book."
+msgstr "No tienes ninguna actividad de lectura para este libro."
+
+#: bookwyrm/templates/book.html:164
+msgid "Create"
+msgstr "Crear"
+
+#: bookwyrm/templates/book.html:186
+msgid "Tags"
+msgstr "Etiquetas"
+
+#: bookwyrm/templates/book.html:190 bookwyrm/templates/snippets/tag.html:18
+msgid "Add tag"
+msgstr "Agregar etiqueta"
+
+#: bookwyrm/templates/book.html:207
+msgid "Subjects"
+msgstr "Sujetos"
+
+#: bookwyrm/templates/book.html:218
+msgid "Places"
+msgstr "Lugares"
+
+#: bookwyrm/templates/book.html:229 bookwyrm/templates/layout.html:64
+#: bookwyrm/templates/lists/lists.html:4 bookwyrm/templates/lists/lists.html:9
+#: bookwyrm/templates/search_results.html:92
+#: bookwyrm/templates/user/user_layout.html:62
+msgid "Lists"
+msgstr "Listas"
+
+#: bookwyrm/templates/book.html:258
+msgid "rated it"
+msgstr "lo calificó con"
+
+#: bookwyrm/templates/components/inline_form.html:8
+#: bookwyrm/templates/feed/feed_layout.html:54
+msgid "Close"
+msgstr "Cerrar"
+
+#: bookwyrm/templates/discover/about.html:7
+#, python-format
+msgid "About %(site_name)s"
+msgstr "Sobre %(site_name)s"
+
+#: bookwyrm/templates/discover/about.html:10
+#: bookwyrm/templates/discover/about.html:20
+msgid "Code of Conduct"
+msgstr "Código de conducta"
+
+#: bookwyrm/templates/discover/about.html:13
+#: bookwyrm/templates/discover/about.html:29
+msgid "Privacy Policy"
+msgstr "Política de privacidad"
+
+#: bookwyrm/templates/discover/discover.html:6
+msgid "Recent Books"
+msgstr "Libros recientes"
+
+#: bookwyrm/templates/discover/landing_layout.html:5
+msgid "Welcome"
+msgstr "Bienvenidos"
+
+#: bookwyrm/templates/discover/landing_layout.html:17
+msgid "Decentralized"
+msgstr "Descentralizado"
+
+#: bookwyrm/templates/discover/landing_layout.html:23
+msgid "Friendly"
+msgstr "Amigable"
+
+#: bookwyrm/templates/discover/landing_layout.html:29
+msgid "Anti-Corporate"
+msgstr "Anti-corporativo"
+
+#: bookwyrm/templates/discover/landing_layout.html:44
+#, python-format
+msgid "Join %(name)s"
+msgstr "Unirse con %(name)s"
+
+#: bookwyrm/templates/discover/landing_layout.html:49
+#: bookwyrm/templates/login.html:48
+msgid "This instance is closed"
+msgstr "Esta instancia está cerrada."
+
+#: bookwyrm/templates/discover/landing_layout.html:55
+msgid "Your Account"
+msgstr "Tu cuenta"
+
+#: bookwyrm/templates/edit_author.html:13 bookwyrm/templates/edit_book.html:13
+msgid "Added:"
+msgstr "Agregado:"
+
+#: bookwyrm/templates/edit_author.html:14 bookwyrm/templates/edit_book.html:14
+msgid "Updated:"
+msgstr "Actualizado:"
+
+#: bookwyrm/templates/edit_author.html:15 bookwyrm/templates/edit_book.html:15
+msgid "Last edited by:"
+msgstr "Editado más recientemente por:"
+
+#: bookwyrm/templates/edit_author.html:31 bookwyrm/templates/edit_book.html:30
+msgid "Metadata"
+msgstr "Metadatos"
+
+#: bookwyrm/templates/edit_author.html:32 bookwyrm/templates/lists/form.html:8
+#: bookwyrm/templates/user/create_shelf_form.html:13
+#: bookwyrm/templates/user/edit_shelf_form.html:14
+msgid "Name:"
+msgstr "Nombre:"
+
+#: bookwyrm/templates/edit_author.html:37
+msgid "Bio:"
+msgstr "Bio:"
+
+#: bookwyrm/templates/edit_author.html:42
+msgid "Wikipedia link:"
+msgstr "Enlace de Wikipedia:"
+
+#: bookwyrm/templates/edit_author.html:47
+msgid "Birth date:"
+msgstr "Fecha de nacimiento:"
+
+#: bookwyrm/templates/edit_author.html:52
+msgid "Death date:"
+msgstr "Fecha de muerte:"
+
+#: bookwyrm/templates/edit_author.html:58
+msgid "Author Identifiers"
+msgstr "Identificadores de autor/autora"
+
+#: bookwyrm/templates/edit_author.html:59 bookwyrm/templates/edit_book.html:103
+msgid "Openlibrary key:"
+msgstr "Clave OpenLibrary:"
+
+#: bookwyrm/templates/edit_author.html:64
+msgid "Librarything key:"
+msgstr "Clave Librarything:"
+
+#: bookwyrm/templates/edit_author.html:69
+msgid "Goodreads key:"
+msgstr "Clave Goodreads:"
+
+#: bookwyrm/templates/edit_book.html:31
+msgid "Title:"
+msgstr "Título:"
+
+#: bookwyrm/templates/edit_book.html:35
+msgid "Subtitle:"
+msgstr "Subtítulo:"
+
+#: bookwyrm/templates/edit_book.html:43
+msgid "Series:"
+msgstr "Serie:"
+
+#: bookwyrm/templates/edit_book.html:47
+msgid "Series number:"
+msgstr "Número de serie:"
+
+#: bookwyrm/templates/edit_book.html:51
+msgid "First published date:"
+msgstr "Fecha de primera publicación:"
+
+#: bookwyrm/templates/edit_book.html:55
+msgid "Published date:"
+msgstr "Fecha de publicación:"
+
+#: bookwyrm/templates/edit_book.html:68
+#: bookwyrm/templates/snippets/shelf.html:9
+msgid "Cover"
+msgstr "Portada:"
+
+#: bookwyrm/templates/edit_book.html:78
+msgid "Physical Properties"
+msgstr "Propiedades físicas:"
+
+#: bookwyrm/templates/edit_book.html:79
+msgid "Format:"
+msgstr "Formato:"
+
+#: bookwyrm/templates/edit_book.html:87
+msgid "Pages:"
+msgstr "Páginas:"
+
+#: bookwyrm/templates/edit_book.html:94
+msgid "Book Identifiers"
+msgstr "Identificadores de libro"
+
+#: bookwyrm/templates/edit_book.html:95
+msgid "ISBN 13:"
+msgstr "ISBN 13:"
+
+#: bookwyrm/templates/edit_book.html:99
+msgid "ISBN 10:"
+msgstr "ISBN 10:"
+
+#: bookwyrm/templates/editions.html:5
+#, python-format
+msgid "Editions of %(book_title)s"
+msgstr "Ediciones de %(book_title)s"
+
+#: bookwyrm/templates/editions.html:9
+#, python-format
+msgid "Editions of \"%(work_title)s\" "
+msgstr "Ediciones de \"%(work_title)s\" "
+
+#: bookwyrm/templates/error.html:4
+msgid "Oops!"
+msgstr "¡Úps!"
+
+#: bookwyrm/templates/error.html:8
+msgid "Server Error"
+msgstr "Error de servidor"
+
+#: bookwyrm/templates/error.html:9
+msgid "Something went wrong! Sorry about that."
+msgstr "¡Algo salió mal! Disculpa."
+
+#: bookwyrm/templates/feed/direct_messages.html:8
+#, python-format
+msgid "Direct Messages with %(username)s "
+msgstr "Mensajes directos con %(username)s "
+
+#: bookwyrm/templates/feed/direct_messages.html:10
+#: bookwyrm/templates/layout.html:87
+msgid "Direct Messages"
+msgstr "Mensajes directos"
+
+#: bookwyrm/templates/feed/direct_messages.html:13
+msgid "All messages"
+msgstr "Todos los mensajes"
+
+#: bookwyrm/templates/feed/direct_messages.html:22
+msgid "You have no messages right now."
+msgstr "No tienes ningún mensaje en este momento."
+
+#: bookwyrm/templates/feed/feed.html:6
+#, python-format
+msgid "%(tab_title)s Timeline"
+msgstr "%(tab_title)s Línea temporal"
+
+#: bookwyrm/templates/feed/feed.html:10 bookwyrm/views/feed.py:33
+msgid "Home"
+msgstr "Hogar"
+
+#: bookwyrm/templates/feed/feed.html:13 bookwyrm/views/feed.py:37
+msgid "Local"
+msgstr "Local"
+
+#: bookwyrm/templates/feed/feed.html:16 bookwyrm/views/feed.py:41
+msgid "Federated"
+msgstr "Federalizado"
+
+#: bookwyrm/templates/feed/feed.html:24
+msgid "Announcements"
+msgstr "Anuncios"
+
+#: bookwyrm/templates/feed/feed.html:32
+msgid "There aren't any activities right now! Try following a user to get started"
+msgstr "¡No hay actividades en este momento! Sigue a otro usuario para empezar"
+
+#: bookwyrm/templates/feed/feed_layout.html:5
+msgid "Updates"
+msgstr "Actualizaciones"
+
+#: bookwyrm/templates/feed/feed_layout.html:11
+msgid "Your books"
+msgstr "Tus libros"
+
+#: bookwyrm/templates/feed/feed_layout.html:13
+msgid "There are no books here right now! Try searching for a book to get started"
+msgstr "¡No hay ningún libro aqui ahorita! Busca a un libro para empezar"
+
+#: bookwyrm/templates/feed/feed_layout.html:23
+#: bookwyrm/templates/user/shelf.html:24
+msgid "To Read"
+msgstr "Para leer"
+
+#: bookwyrm/templates/feed/feed_layout.html:24
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Currently Reading"
+msgstr "Leyendo actualmente"
+
+#: bookwyrm/templates/feed/feed_layout.html:25
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:11
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Read"
+msgstr "Leer"
+
+#: bookwyrm/templates/feed/feed_layout.html:76 bookwyrm/templates/goal.html:26
+#: bookwyrm/templates/snippets/goal_card.html:6
+#, python-format
+msgid "%(year)s Reading Goal"
+msgstr "%(year)s Meta de lectura"
+
+#: bookwyrm/templates/feed/status.html:8
+msgid "Back"
+msgstr "Volver"
+
+#: bookwyrm/templates/goal.html:7
+#, python-format
+msgid "%(year)s Reading Progress"
+msgstr "%(year)s Progreso de la meta de lectura"
+
+#: bookwyrm/templates/goal.html:11
+msgid "Edit Goal"
+msgstr "Editar meta"
+
+#: bookwyrm/templates/goal.html:30
+#: bookwyrm/templates/snippets/goal_card.html:13
+#, python-format
+msgid "Set a goal for how many books you'll finish reading in %(year)s, and track your progress throughout the year."
+msgstr "Establece una meta para cuantos libros leerás en %(year)s, y seguir tu progreso durante el año."
+
+#: bookwyrm/templates/goal.html:39
+#, python-format
+msgid "%(name)s hasn't set a reading goal for %(year)s."
+msgstr "%(name)s no ha establecido una meta de lectura para %(year)s."
+
+#: bookwyrm/templates/goal.html:51
+#, python-format
+msgid "Your %(year)s Books"
+msgstr "Tus libros de %(year)s"
+
+#: bookwyrm/templates/goal.html:53
+#, python-format
+msgid "%(username)s's %(year)s Books"
+msgstr "Los libros de %(username)s para %(year)s"
+
+#: bookwyrm/templates/import.html:5 bookwyrm/templates/import.html:9
+#: bookwyrm/templates/layout.html:102
+msgid "Import Books"
+msgstr "Importar libros"
+
+#: bookwyrm/templates/import.html:14
+msgid "Data source"
+msgstr "Fuente de datos"
+
+#: bookwyrm/templates/import.html:32
+msgid "Include reviews"
+msgstr "Incluir reseñas"
+
+#: bookwyrm/templates/import.html:37
+msgid "Privacy setting for imported reviews:"
+msgstr "Configuración de privacidad para las reseñas importadas:"
+
+#: bookwyrm/templates/import.html:41
+msgid "Import"
+msgstr "Importar"
+
+#: bookwyrm/templates/import.html:46
+msgid "Recent Imports"
+msgstr "Importaciones recientes"
+
+#: bookwyrm/templates/import.html:48
+msgid "No recent imports"
+msgstr "No hay ninguna importación reciente"
+
+#: bookwyrm/templates/import_status.html:6
+#: bookwyrm/templates/import_status.html:10
+msgid "Import Status"
+msgstr "Status de importación"
+
+#: bookwyrm/templates/import_status.html:13
+msgid "Import started:"
+msgstr "Importación ha empezado:"
+
+#: bookwyrm/templates/import_status.html:17
+msgid "Import completed:"
+msgstr "Importación ha terminado:"
+
+#: bookwyrm/templates/import_status.html:20
+msgid "TASK FAILED"
+msgstr "TAREA FALLÓ"
+
+#: bookwyrm/templates/import_status.html:26
+msgid "Import still in progress."
+msgstr "Importación todavia en progreso"
+
+#: bookwyrm/templates/import_status.html:28
+msgid "(Hit reload to update!)"
+msgstr "(¡Refresca para actualizar!)"
+
+#: bookwyrm/templates/import_status.html:35
+msgid "Failed to load"
+msgstr "Se falló a cargar"
+
+#: bookwyrm/templates/import_status.html:44
+#, python-format
+msgid "Jump to the bottom of the list to select the %(failed_count)s items which failed to import."
+msgstr ""
+
+#: bookwyrm/templates/import_status.html:79
+msgid "Select all"
+msgstr "Seleccionar todo"
+
+#: bookwyrm/templates/import_status.html:82
+msgid "Retry items"
+msgstr "Reintentar ítems"
+
+#: bookwyrm/templates/import_status.html:108
+msgid "Successfully imported"
+msgstr "Importado exitosamente"
+
+#: bookwyrm/templates/import_status.html:112
+#: bookwyrm/templates/lists/curate.html:14
+msgid "Book"
+msgstr "Libro"
+
+#: bookwyrm/templates/import_status.html:115
+#: bookwyrm/templates/snippets/create_status_form.html:10
+#: bookwyrm/templates/snippets/shelf.html:10
+msgid "Title"
+msgstr "Título"
+
+#: bookwyrm/templates/import_status.html:118
+#: bookwyrm/templates/snippets/shelf.html:11
+msgid "Author"
+msgstr "Autor/Autora"
+
+#: bookwyrm/templates/import_status.html:141
+msgid "Imported"
+msgstr "Importado"
+
+#: bookwyrm/templates/invite.html:4 bookwyrm/templates/invite.html:12
+#: bookwyrm/templates/login.html:43
+msgid "Create an Account"
+msgstr "Crear una cuenta"
+
+#: bookwyrm/templates/invite.html:21
+msgid "Permission Denied"
+msgstr "Permiso denegado"
+
+#: bookwyrm/templates/invite.html:22
+msgid "Sorry! This invite code is no longer valid."
+msgstr "¡Disculpa! Este código de invitación no queda válido."
+
+#: bookwyrm/templates/isbn_search_results.html:4
+#: bookwyrm/templates/search_results.html:4
+msgid "Search Results"
+msgstr "Resultados de búsqueda"
+
+#: bookwyrm/templates/isbn_search_results.html:9
+#: bookwyrm/templates/search_results.html:9
+#, python-format
+msgid "Search Results for \"%(query)s\""
+msgstr "Resultados de búsqueda por \"%(query)s\""
+
+#: bookwyrm/templates/isbn_search_results.html:14
+#: bookwyrm/templates/search_results.html:14
+msgid "Matching Books"
+msgstr "Libros correspondientes"
+
+#: bookwyrm/templates/isbn_search_results.html:17
+#: bookwyrm/templates/search_results.html:17
+#, python-format
+msgid "No books found for \"%(query)s\""
+msgstr "No se encontró ningún libro correspondiente a \"%(query)s\""
+
+#: bookwyrm/templates/layout.html:33
+msgid "Search for a book or user"
+msgstr "Buscar un libro o un usuario"
+
+#: bookwyrm/templates/layout.html:37 bookwyrm/templates/layout.html:38
+#: bookwyrm/templates/lists/list.html:62
+msgid "Search"
+msgstr "Buscar"
+
+#: bookwyrm/templates/layout.html:47 bookwyrm/templates/layout.html:48
+msgid "Main navigation menu"
+msgstr "Menú de navigación central"
+
+#: bookwyrm/templates/layout.html:58
+msgid "Your shelves"
+msgstr "Tus estantes"
+
+#: bookwyrm/templates/layout.html:61
+msgid "Feed"
+msgstr "Actividad"
+
+#: bookwyrm/templates/layout.html:92
+#: bookwyrm/templates/preferences/preferences_layout.html:14
+msgid "Profile"
+msgstr "Perfil"
+
+#: bookwyrm/templates/layout.html:97
+msgid "Settings"
+msgstr "Configuración"
+
+#: bookwyrm/templates/layout.html:111
+#: bookwyrm/templates/settings/admin_layout.html:19
+#: bookwyrm/templates/settings/manage_invites.html:3
+msgid "Invites"
+msgstr "Invitaciones"
+
+#: bookwyrm/templates/layout.html:118
+msgid "Site Configuration"
+msgstr "Configuracion de sitio"
+
+#: bookwyrm/templates/layout.html:125
+msgid "Log out"
+msgstr "Cerrar sesión"
+
+#: bookwyrm/templates/layout.html:133 bookwyrm/templates/layout.html:134
+#: bookwyrm/templates/notifications.html:6
+#: bookwyrm/templates/notifications.html:10
+msgid "Notifications"
+msgstr "Notificaciones"
+
+#: bookwyrm/templates/layout.html:151 bookwyrm/templates/layout.html:155
+#: bookwyrm/templates/login.html:17
+#: bookwyrm/templates/snippets/register_form.html:4
+msgid "Username:"
+msgstr "Nombre de usuario:"
+
+#: bookwyrm/templates/layout.html:156
+msgid "password"
+msgstr "contraseña"
+
+#: bookwyrm/templates/layout.html:157 bookwyrm/templates/login.html:36
+msgid "Forgot your password?"
+msgstr "¿Olvidaste tu contraseña?"
+
+#: bookwyrm/templates/layout.html:160 bookwyrm/templates/login.html:10
+#: bookwyrm/templates/login.html:33
+msgid "Log in"
+msgstr "Iniciar sesión"
+
+#: bookwyrm/templates/layout.html:168
+msgid "Join"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:191
+msgid "About this server"
+msgstr "Sobre este servidor"
+
+#: bookwyrm/templates/layout.html:195
+msgid "Contact site admin"
+msgstr "Contactarse con administradores del sitio"
+
+#: bookwyrm/templates/layout.html:202
+#, python-format
+msgid "Support %(site_name)s on %(support_title)s "
+msgstr ""
+
+#: bookwyrm/templates/layout.html:206
+msgid "BookWyrm is open source software. You can contribute or report issues on GitHub ."
+msgstr "BookWyrm es software de código abierto. Puedes contribuir o reportar problemas en GitHub ."
+
+#: bookwyrm/templates/lists/create_form.html:5
+#: bookwyrm/templates/lists/lists.html:17
+msgid "Create List"
+msgstr "Crear lista"
+
+#: bookwyrm/templates/lists/created_text.html:5
+#, fuzzy, python-format
+#| msgid "Added by %(username)s "
+msgid "Created and curated by %(username)s "
+msgstr "Agregado por %(username)s "
+
+#: bookwyrm/templates/lists/created_text.html:7
+#, fuzzy, python-format
+#| msgid "Added by %(username)s "
+msgid "Created by %(username)s "
+msgstr "Agregado por %(username)s "
+
+#: bookwyrm/templates/lists/curate.html:6
+msgid "Pending Books"
+msgstr "Libros pendientes"
+
+#: bookwyrm/templates/lists/curate.html:7
+msgid "Go to list"
+msgstr "Irse a lista"
+
+#: bookwyrm/templates/lists/curate.html:9
+msgid "You're all set!"
+msgstr "!Está todo listo¡"
+
+#: bookwyrm/templates/lists/curate.html:15
+msgid "Suggested by"
+msgstr "Sugerido por"
+
+#: bookwyrm/templates/lists/curate.html:35
+msgid "Approve"
+msgstr "Aprobar"
+
+#: bookwyrm/templates/lists/curate.html:41
+msgid "Discard"
+msgstr "Desechar"
+
+#: bookwyrm/templates/lists/edit_form.html:5
+#: bookwyrm/templates/lists/list_layout.html:18
+msgid "Edit List"
+msgstr "Editar lista"
+
+#: bookwyrm/templates/lists/form.html:18
+msgid "List curation:"
+msgstr "Enumerar lista de comisariado:"
+
+#: bookwyrm/templates/lists/form.html:21
+msgid "Closed"
+msgstr "Cerrado"
+
+#: bookwyrm/templates/lists/form.html:22
+msgid "Only you can add and remove books to this list"
+msgstr "Solo tú puedes agregar a y sacar libros de esta lista"
+
+#: bookwyrm/templates/lists/form.html:26
+msgid "Curated"
+msgstr "De comisariado"
+
+#: bookwyrm/templates/lists/form.html:27
+msgid "Anyone can suggest books, subject to your approval"
+msgstr "Cualquier usuario puede sugerir libros, en cuanto lo hayas aprobado"
+
+#: bookwyrm/templates/lists/form.html:31
+msgid "Open"
+msgstr "Abierto"
+
+#: bookwyrm/templates/lists/form.html:32
+msgid "Anyone can add books to this list"
+msgstr "Cualquer usuario puede agregar libros a esta lista"
+
+#: bookwyrm/templates/lists/list.html:17
+msgid "This list is currently empty"
+msgstr "Esta lista está vacia"
+
+#: bookwyrm/templates/lists/list.html:35
+#, python-format
+msgid "Added by %(username)s "
+msgstr "Agregado por %(username)s "
+
+#: bookwyrm/templates/lists/list.html:41
+msgid "Remove"
+msgstr "Quitar"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Add Books"
+msgstr "Agregar libros"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Suggest Books"
+msgstr "Sugerir libros"
+
+#: bookwyrm/templates/lists/list.html:58
+msgid "Search for a book"
+msgstr "Buscar libros"
+
+#: bookwyrm/templates/lists/list.html:63
+msgid "search"
+msgstr "buscar"
+
+#: bookwyrm/templates/lists/list.html:69
+msgid "Clear search"
+msgstr "Borrar búsqueda"
+
+#: bookwyrm/templates/lists/list.html:74
+#, python-format
+msgid "No books found matching the query \"%(query)s\""
+msgstr "No se encontró ningún libro correspondiente a la búsqueda: \"%(query)s\""
+
+#: bookwyrm/templates/lists/list.html:75
+msgid "No books found"
+msgstr "No se encontró ningún libro"
+
+#: bookwyrm/templates/lists/list.html:89
+msgid "Suggest"
+msgstr "Sugerir"
+
+#: bookwyrm/templates/lists/lists.html:14
+msgid "Your lists"
+msgstr "Tus listas"
+
+#: bookwyrm/templates/lists/lists.html:32
+#, python-format
+msgid "See all %(size)s lists"
+msgstr "Ver las %(size)s listas"
+
+#: bookwyrm/templates/lists/lists.html:40
+msgid "Recent Lists"
+msgstr "Listas recientes"
+
+#: bookwyrm/templates/login.html:4
+msgid "Login"
+msgstr "Iniciar sesión"
+
+#: bookwyrm/templates/login.html:23 bookwyrm/templates/password_reset.html:17
+#: bookwyrm/templates/snippets/register_form.html:22
+msgid "Password:"
+msgstr "Contraseña:"
+
+#: bookwyrm/templates/login.html:49
+msgid "Contact an administrator to get an invite"
+msgstr "Contactar a unx administradorx para recibir una invitación"
+
+#: bookwyrm/templates/login.html:59
+msgid "More about this site"
+msgstr "Más sobre este sitio"
+
+#: bookwyrm/templates/notfound.html:4 bookwyrm/templates/notfound.html:8
+msgid "Not Found"
+msgstr "No encontrado"
+
+#: bookwyrm/templates/notfound.html:9
+msgid "The page you requested doesn't seem to exist!"
+msgstr "¡Parece que la página solicitada no existe!"
+
+#: bookwyrm/templates/notifications.html:14
+msgid "Delete notifications"
+msgstr "Borrar notificaciones"
+
+#: bookwyrm/templates/notifications.html:51
+#, python-format
+msgid "favorited your review of %(book_title)s "
+msgstr "le gustó tu reseña de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:53
+#, python-format
+msgid "favorited your comment on %(book_title)s "
+msgstr "le gustó tu comentario en %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:55
+#, python-format
+msgid "favorited your quote from %(book_title)s "
+msgstr "le gustó tu cita de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:57
+#, python-format
+msgid "favorited your status "
+msgstr "le gustó tu status "
+
+#: bookwyrm/templates/notifications.html:62
+#, python-format
+msgid "mentioned you in a review of %(book_title)s "
+msgstr "te mencionó en una reseña de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:64
+#, python-format
+msgid "mentioned you in a comment on %(book_title)s "
+msgstr "te mencionó en un comentario de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:66
+#, python-format
+msgid "mentioned you in a quote from %(book_title)s "
+msgstr "te mencionó en una cita de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:68
+#, python-format
+msgid "mentioned you in a status "
+msgstr "te mencionó en un status "
+
+#: bookwyrm/templates/notifications.html:73
+#, python-format
+msgid "replied to your review of %(book_title)s "
+msgstr "respondió a tu reseña de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:75
+#, python-format
+msgid "replied to your comment on %(book_title)s "
+msgstr "respondió a tu comentario en %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:77
+#, python-format
+msgid "replied to your quote from %(book_title)s "
+msgstr "respondió a tu cita de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:79
+#, python-format
+msgid "replied to your status "
+msgstr "respondió a tu status "
+
+#: bookwyrm/templates/notifications.html:83
+msgid "followed you"
+msgstr "te siguió"
+
+#: bookwyrm/templates/notifications.html:86
+msgid "sent you a follow request"
+msgstr "te quiere seguir"
+
+#: bookwyrm/templates/notifications.html:92
+#, python-format
+msgid "boosted your review of %(book_title)s "
+msgstr "respaldó tu reseña de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:94
+#, python-format
+msgid "boosted your comment on%(book_title)s "
+msgstr "respaldó tu comentario en%(book_title)s "
+
+#: bookwyrm/templates/notifications.html:96
+#, python-format
+msgid "boosted your quote from %(book_title)s "
+msgstr "respaldó tucita de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:98
+#, python-format
+msgid "boosted your status "
+msgstr "respaldó tu status "
+
+#: bookwyrm/templates/notifications.html:102
+#, python-format
+msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgstr " agregó %(book_title)s a tu lista \"%(list_name)s \""
+
+#: bookwyrm/templates/notifications.html:104
+#, python-format
+msgid " suggested adding %(book_title)s to your list \"%(list_name)s \""
+msgstr " sugirió agregar %(book_title)s a tu lista \"%(list_name)s \""
+
+#: bookwyrm/templates/notifications.html:108
+#, python-format
+msgid " your import completed."
+msgstr " tu importación ha terminado."
+
+#: bookwyrm/templates/notifications.html:142
+msgid "You're all caught up!"
+msgstr "¡Estás al día!"
+
+#: bookwyrm/templates/password_reset.html:4
+#: bookwyrm/templates/password_reset.html:10
+#: bookwyrm/templates/password_reset_request.html:4
+#: bookwyrm/templates/password_reset_request.html:10
+msgid "Reset Password"
+msgstr "Restablecer contraseña"
+
+#: bookwyrm/templates/password_reset.html:23
+#: bookwyrm/templates/preferences/change_password.html:18
+msgid "Confirm password:"
+msgstr "Confirmar contraseña:"
+
+#: bookwyrm/templates/password_reset.html:30
+msgid "Confirm"
+msgstr "Confirmar"
+
+#: bookwyrm/templates/password_reset_request.html:12
+msgid "A link to reset your password will be sent to your email address"
+msgstr "Un enlace para restablecer tu contraseña se enviará a tu dirección de correo electrónico"
+
+#: bookwyrm/templates/password_reset_request.html:16
+#: bookwyrm/templates/preferences/edit_user.html:38
+#: bookwyrm/templates/snippets/register_form.html:13
+msgid "Email address:"
+msgstr "Dirección de correo electrónico:"
+
+#: bookwyrm/templates/password_reset_request.html:23
+msgid "Reset password"
+msgstr "Restablecer contraseña"
+
+#: bookwyrm/templates/preferences/blocks.html:4
+#: bookwyrm/templates/preferences/blocks.html:7
+#: bookwyrm/templates/preferences/preferences_layout.html:23
+msgid "Blocked Users"
+msgstr "Usuarios bloqueados"
+
+#: bookwyrm/templates/preferences/blocks.html:12
+msgid "No users currently blocked."
+msgstr "No hay ningún usuario bloqueado actualmente."
+
+#: bookwyrm/templates/preferences/change_password.html:4
+#: bookwyrm/templates/preferences/change_password.html:7
+#: bookwyrm/templates/preferences/change_password.html:21
+#: bookwyrm/templates/preferences/preferences_layout.html:17
+msgid "Change Password"
+msgstr "Cambiar contraseña"
+
+#: bookwyrm/templates/preferences/change_password.html:14
+msgid "New password:"
+msgstr "Nueva contraseña:"
+
+#: bookwyrm/templates/preferences/edit_user.html:4
+#: bookwyrm/templates/preferences/edit_user.html:7
+msgid "Edit Profile"
+msgstr "Editar perfil"
+
+#: bookwyrm/templates/preferences/edit_user.html:17
+msgid "Avatar:"
+msgstr "Avatar:"
+
+#: bookwyrm/templates/preferences/edit_user.html:24
+msgid "Display name:"
+msgstr "Nombre de visualización:"
+
+#: bookwyrm/templates/preferences/edit_user.html:31
+msgid "Summary:"
+msgstr "Resumen:"
+
+#: bookwyrm/templates/preferences/edit_user.html:46
+msgid "Manually approve followers:"
+msgstr "Aprobar seguidores a mano:"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:11
+msgid "Account"
+msgstr "Cuenta"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:20
+msgid "Relationships"
+msgstr "Relaciones"
+
+#: bookwyrm/templates/search_results.html:33
+msgid "Didn't find what you were looking for?"
+msgstr "¿No encontraste lo que buscabas?"
+
+#: bookwyrm/templates/search_results.html:35
+msgid "Show results from other catalogues"
+msgstr "Mostrar resultados de otros catálogos"
+
+#: bookwyrm/templates/search_results.html:57
+msgid "Import book"
+msgstr "Importar libro"
+
+#: bookwyrm/templates/search_results.html:67
+msgid "Hide results from other catalogues"
+msgstr "Ocultar resultados de otros catálogos"
+
+#: bookwyrm/templates/search_results.html:75
+msgid "Matching Users"
+msgstr "Usuarios correspondientes"
+
+#: bookwyrm/templates/search_results.html:77
+#, python-format
+msgid "No users found for \"%(query)s\""
+msgstr "No se encontró ningún usuario correspondiente a \"%(query)s\""
+
+#: bookwyrm/templates/search_results.html:94
+#, python-format
+msgid "No lists found for \"%(query)s\""
+msgstr "No se encontró ningúna lista correspondiente a \"%(query)s\""
+
+#: bookwyrm/templates/settings/admin_layout.html:4
+msgid "Administration"
+msgstr "Adminstración"
+
+#: bookwyrm/templates/settings/admin_layout.html:15
+msgid "Manage Users"
+msgstr "Administrar usuarios"
+
+#: bookwyrm/templates/settings/admin_layout.html:23
+#: bookwyrm/templates/settings/federation.html:4
+msgid "Federated Servers"
+msgstr "Servidores federalizados"
+
+#: bookwyrm/templates/settings/admin_layout.html:28
+msgid "Instance Settings"
+msgstr "Configuración de instancia"
+
+#: bookwyrm/templates/settings/admin_layout.html:32
+#: bookwyrm/templates/settings/site.html:4
+#: bookwyrm/templates/settings/site.html:6
+msgid "Site Settings"
+msgstr "Configuración de sitio"
+
+#: bookwyrm/templates/settings/admin_layout.html:35
+#: bookwyrm/templates/settings/site.html:13
+msgid "Instance Info"
+msgstr "Información de instancia"
+
+#: bookwyrm/templates/settings/admin_layout.html:36
+#: bookwyrm/templates/settings/site.html:39
+msgid "Images"
+msgstr "Imagenes"
+
+#: bookwyrm/templates/settings/admin_layout.html:37
+#: bookwyrm/templates/settings/site.html:59
+msgid "Footer Content"
+msgstr "Contenido del pie de página"
+
+#: bookwyrm/templates/settings/admin_layout.html:38
+#: bookwyrm/templates/settings/site.html:77
+msgid "Registration"
+msgstr "Registración"
+
+#: bookwyrm/templates/settings/federation.html:10
+msgid "Server name"
+msgstr "Nombre de servidor"
+
+#: bookwyrm/templates/settings/federation.html:11
+msgid "Software"
+msgstr "Software"
+
+#: bookwyrm/templates/settings/federation.html:12
+msgid "Status"
+msgstr "Status"
+
+#: bookwyrm/templates/settings/manage_invites.html:7
+msgid "Generate New Invite"
+msgstr "Generar nuevo invitación"
+
+#: bookwyrm/templates/settings/manage_invites.html:13
+msgid "Expiry:"
+msgstr "Vencimiento:"
+
+#: bookwyrm/templates/settings/manage_invites.html:19
+msgid "Use limit:"
+msgstr "Límite de uso:"
+
+#: bookwyrm/templates/settings/manage_invites.html:26
+msgid "Create Invite"
+msgstr "Crear invitación"
+
+#: bookwyrm/templates/settings/manage_invites.html:33
+msgid "Link"
+msgstr "Enlace"
+
+#: bookwyrm/templates/settings/manage_invites.html:34
+msgid "Expires"
+msgstr "Vence"
+
+#: bookwyrm/templates/settings/manage_invites.html:35
+msgid "Max uses"
+msgstr "Número máximo de usos"
+
+#: bookwyrm/templates/settings/manage_invites.html:36
+msgid "Times used"
+msgstr "Número de usos"
+
+#: bookwyrm/templates/settings/manage_invites.html:39
+msgid "No active invites"
+msgstr "No invitaciónes activas"
+
+#: bookwyrm/templates/settings/site.html:15
+msgid "Instance Name:"
+msgstr "Nombre de instancia:"
+
+#: bookwyrm/templates/settings/site.html:19
+msgid "Tagline:"
+msgstr "Lema:"
+
+#: bookwyrm/templates/settings/site.html:23
+msgid "Instance description:"
+msgstr "Descripción de instancia:"
+
+#: bookwyrm/templates/settings/site.html:27
+msgid "Code of conduct:"
+msgstr "Código de conducta:"
+
+#: bookwyrm/templates/settings/site.html:31
+msgid "Privacy Policy:"
+msgstr "Política de privacidad:"
+
+#: bookwyrm/templates/settings/site.html:42
+msgid "Logo:"
+msgstr "Logo:"
+
+#: bookwyrm/templates/settings/site.html:46
+msgid "Logo small:"
+msgstr "Logo pequeño:"
+
+#: bookwyrm/templates/settings/site.html:50
+msgid "Favicon:"
+msgstr "Favicon:"
+
+#: bookwyrm/templates/settings/site.html:61
+msgid "Support link:"
+msgstr "Enlace de apoyo:"
+
+#: bookwyrm/templates/settings/site.html:65
+msgid "Support title:"
+msgstr "Título de apoyo:"
+
+#: bookwyrm/templates/settings/site.html:69
+msgid "Admin email:"
+msgstr "Correo electrónico de administradorx:"
+
+#: bookwyrm/templates/settings/site.html:79
+msgid "Allow registration:"
+msgstr "Permitir registración:"
+
+#: bookwyrm/templates/settings/site.html:83
+msgid "Registration closed text:"
+msgstr "Texto de registración cerrada:"
+
+#: bookwyrm/templates/snippets/block_button.html:5
+msgid "Block"
+msgstr "Bloquear"
+
+#: bookwyrm/templates/snippets/block_button.html:10
+msgid "Un-block"
+msgstr "Desbloquear"
+
+#: bookwyrm/templates/snippets/book_titleby.html:3
+#, python-format
+msgid "%(title)s by "
+msgstr "%(title)s por "
+
+#: bookwyrm/templates/snippets/boost_button.html:8
+#: bookwyrm/templates/snippets/boost_button.html:9
+#: bookwyrm/templates/snippets/status/status_body.html:41
+#: bookwyrm/templates/snippets/status/status_body.html:42
+msgid "Boost status"
+msgstr "Status de respaldo"
+
+#: bookwyrm/templates/snippets/boost_button.html:16
+#: bookwyrm/templates/snippets/boost_button.html:17
+msgid "Un-boost status"
+msgstr "Status de des-respaldo"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:3
+msgid "Spoiler alert:"
+msgstr "Alerta de spoiler:"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:4
+msgid "Spoilers ahead!"
+msgstr "¡Advertencia, ya vienen spoilers!"
+
+#: bookwyrm/templates/snippets/create_status.html:9
+msgid "Review"
+msgstr "Reseña"
+
+#: bookwyrm/templates/snippets/create_status.html:12
+#: bookwyrm/templates/snippets/create_status_form.html:44
+msgid "Comment"
+msgstr "Comentario"
+
+#: bookwyrm/templates/snippets/create_status.html:15
+msgid "Quote"
+msgstr "Cita"
+
+#: bookwyrm/templates/snippets/create_status_form.html:21
+#: bookwyrm/templates/snippets/shelf.html:17
+msgid "Rating"
+msgstr "Calificación"
+
+#: bookwyrm/templates/snippets/create_status_form.html:23
+#: bookwyrm/templates/snippets/rate_action.html:14
+#: bookwyrm/templates/snippets/stars.html:3
+msgid "No rating"
+msgstr "No calificación"
+
+#: bookwyrm/templates/snippets/create_status_form.html:54
+msgid "Include spoiler alert"
+msgstr "Incluir alerta de spoiler"
+
+#: bookwyrm/templates/snippets/create_status_form.html:60
+#: bookwyrm/templates/snippets/privacy-icons.html:15
+#: bookwyrm/templates/snippets/privacy-icons.html:16
+#: bookwyrm/templates/snippets/privacy_select.html:19
+msgid "Private"
+msgstr "Privada"
+
+#: bookwyrm/templates/snippets/create_status_form.html:67
+msgid "Post"
+msgstr "Compartir"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:4
+msgid "Delete these read dates?"
+msgstr "¿Eliminar estas fechas de lectura?"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:7
+#, python-format
+msgid "You are deleting this readthrough and its %(count)s associated progress updates."
+msgstr "Estás eliminando esta lectura y sus %(count)s actualizaciones de progreso asociados."
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:15
+#: bookwyrm/templates/snippets/follow_request_buttons.html:13
+msgid "Delete"
+msgstr "Eliminar"
+
+#: bookwyrm/templates/snippets/fav_button.html:7
+#: bookwyrm/templates/snippets/fav_button.html:8
+#: bookwyrm/templates/snippets/status/status_body.html:45
+#: bookwyrm/templates/snippets/status/status_body.html:46
+msgid "Like status"
+msgstr "Me gusta status"
+
+#: bookwyrm/templates/snippets/fav_button.html:15
+#: bookwyrm/templates/snippets/fav_button.html:16
+msgid "Un-like status"
+msgstr "Quitar me gusta de status"
+
+#: bookwyrm/templates/snippets/follow_button.html:6
+msgid "Follow request already sent."
+msgstr "Solicitud de seguidor ya se ha enviado."
+
+#: bookwyrm/templates/snippets/follow_button.html:19
+msgid "Send follow request"
+msgstr "Envia solicitud de seguidor"
+
+#: bookwyrm/templates/snippets/follow_button.html:21
+msgid "Follow"
+msgstr "Seguir"
+
+#: bookwyrm/templates/snippets/follow_button.html:27
+msgid "Unfollow"
+msgstr "Dejar de seguir"
+
+#: bookwyrm/templates/snippets/follow_request_buttons.html:8
+msgid "Accept"
+msgstr "Aceptar"
+
+#: bookwyrm/templates/snippets/generated_status/goal.html:1
+#, python-format
+msgid "set a goal to read %(counter)s book in %(year)s"
+msgid_plural "set a goal to read %(counter)s books in %(year)s"
+msgstr[0] "estableció una meta de leer %(counter)s libro en %(year)s"
+msgstr[1] "estableció una meta de leer %(counter)s libros en %(year)s"
+
+#: bookwyrm/templates/snippets/goal_card.html:21
+msgid "Dismiss message"
+msgstr "Desechar mensaje"
+
+#: bookwyrm/templates/snippets/goal_card.html:22
+#, python-format
+msgid "You can set or change your reading goal any time from your profile page "
+msgstr "Puedes establecer o cambiar tu meta de lectura en cualquier momento que desees desde tu perfil "
+
+#: bookwyrm/templates/snippets/goal_form.html:9
+msgid "Reading goal:"
+msgstr "Meta de lectura:"
+
+#: bookwyrm/templates/snippets/goal_form.html:14
+msgid "books"
+msgstr "libros"
+
+#: bookwyrm/templates/snippets/goal_form.html:19
+msgid "Goal privacy:"
+msgstr "Privacidad de meta:"
+
+#: bookwyrm/templates/snippets/goal_form.html:26
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:37
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:29
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:20
+msgid "Post to feed"
+msgstr "Compartir con tu feed"
+
+#: bookwyrm/templates/snippets/goal_form.html:30
+msgid "Set goal"
+msgstr "Establecer meta"
+
+#: bookwyrm/templates/snippets/goal_progress.html:5
+msgid "Success!"
+msgstr "¡Meta logrado!"
+
+#: bookwyrm/templates/snippets/goal_progress.html:7
+#, python-format
+msgid "%(percent)s%% complete!"
+msgstr "%(percent)s%% terminado!"
+
+#: bookwyrm/templates/snippets/goal_progress.html:10
+#, python-format
+msgid "You've read %(read_count)s of %(goal_count)s books ."
+msgstr "Has leído %(read_count)s de %(goal_count)s libros ."
+
+#: bookwyrm/templates/snippets/goal_progress.html:12
+#, python-format
+msgid "%(username)s has read %(read_count)s of %(goal_count)s books ."
+msgstr "%(username)s ha leído %(read_count)s de %(goal_count)s libros ."
+
+#: bookwyrm/templates/snippets/pagination.html:7
+msgid "Previous"
+msgstr "Anterior"
+
+#: bookwyrm/templates/snippets/pagination.html:15
+msgid "Next"
+msgstr "Siguiente"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:3
+#: bookwyrm/templates/snippets/privacy-icons.html:4
+#: bookwyrm/templates/snippets/privacy_select.html:10
+msgid "Public"
+msgstr "Público"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:7
+#: bookwyrm/templates/snippets/privacy-icons.html:8
+#: bookwyrm/templates/snippets/privacy_select.html:13
+msgid "Unlisted"
+msgstr "Privado"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:12
+msgid "Followers-only"
+msgstr "Solo seguidores"
+
+#: bookwyrm/templates/snippets/privacy_select.html:6
+msgid "Post privacy"
+msgstr "Privacidad de publicación"
+
+#: bookwyrm/templates/snippets/privacy_select.html:16
+#: bookwyrm/templates/user/followers.html:13
+msgid "Followers"
+msgstr "Seguidores"
+
+#: bookwyrm/templates/snippets/progress_update.html:6
+msgid "Progress:"
+msgstr "Progreso:"
+
+#: bookwyrm/templates/snippets/progress_update.html:16
+#: bookwyrm/templates/snippets/readthrough_form.html:22
+msgid "pages"
+msgstr "páginas"
+
+#: bookwyrm/templates/snippets/progress_update.html:17
+#: bookwyrm/templates/snippets/readthrough_form.html:23
+msgid "percent"
+msgstr "por ciento"
+
+#: bookwyrm/templates/snippets/progress_update.html:25
+#, python-format
+msgid "of %(book.pages)s pages"
+msgstr "de %(book.pages)s páginas"
+
+#: bookwyrm/templates/snippets/rate_action.html:4
+msgid "Leave a rating"
+msgstr "Da una calificación"
+
+#: bookwyrm/templates/snippets/rate_action.html:29
+msgid "Rate"
+msgstr "Calificar"
+
+#: bookwyrm/templates/snippets/readthrough.html:7
+msgid "Progress Updates:"
+msgstr "Actualizaciones de progreso:"
+
+#: bookwyrm/templates/snippets/readthrough.html:11
+msgid "finished"
+msgstr "terminado"
+
+#: bookwyrm/templates/snippets/readthrough.html:14
+msgid "Show all updates"
+msgstr "Mostrar todas las actualizaciones"
+
+#: bookwyrm/templates/snippets/readthrough.html:30
+msgid "Delete this progress update"
+msgstr "Eliminar esta actualización de progreso"
+
+#: bookwyrm/templates/snippets/readthrough.html:40
+msgid "started"
+msgstr "empezado"
+
+#: bookwyrm/templates/snippets/readthrough.html:46
+#: bookwyrm/templates/snippets/readthrough.html:60
+msgid "Edit read dates"
+msgstr "Editar fechas de lectura"
+
+#: bookwyrm/templates/snippets/readthrough.html:50
+msgid "Delete these read dates"
+msgstr "Eliminar estas fechas de lectura"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:7
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:19
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:17
+msgid "Started reading"
+msgstr "Lectura se empezó"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:14
+msgid "Progress"
+msgstr "Progreso"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:30
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:25
+msgid "Finished reading"
+msgstr "Lectura se terminó"
+
+#: bookwyrm/templates/snippets/register_form.html:32
+msgid "Sign Up"
+msgstr "Inscribirse"
+
+#: bookwyrm/templates/snippets/rss_title.html:5
+#: bookwyrm/templates/snippets/status/status_header.html:11
+msgid "rated"
+msgstr "calificó"
+
+#: bookwyrm/templates/snippets/rss_title.html:7
+#: bookwyrm/templates/snippets/status/status_header.html:13
+msgid "reviewed"
+msgstr "reseñó"
+
+#: bookwyrm/templates/snippets/rss_title.html:9
+#: bookwyrm/templates/snippets/status/status_header.html:15
+msgid "commented on"
+msgstr "comentó en"
+
+#: bookwyrm/templates/snippets/rss_title.html:11
+#: bookwyrm/templates/snippets/status/status_header.html:17
+msgid "quoted"
+msgstr "citó"
+
+#: bookwyrm/templates/snippets/search_result_text.html:3
+#, python-format
+msgid "by %(author)s"
+msgstr "por %(author)s"
+
+#: bookwyrm/templates/snippets/shelf.html:12
+msgid "Published"
+msgstr "Publicado"
+
+#: bookwyrm/templates/snippets/shelf.html:13
+msgid "Shelved"
+msgstr "Archivado"
+
+#: bookwyrm/templates/snippets/shelf.html:14
+msgid "Started"
+msgstr "Empezado"
+
+#: bookwyrm/templates/snippets/shelf.html:15
+msgid "Finished"
+msgstr "Terminado"
+
+#: bookwyrm/templates/snippets/shelf.html:16
+msgid "External links"
+msgstr "Enlaces externos"
+
+#: bookwyrm/templates/snippets/shelf.html:44
+msgid "OpenLibrary"
+msgstr "OpenLibrary"
+
+#: bookwyrm/templates/snippets/shelf.html:61
+msgid "This shelf is empty."
+msgstr "Este estante está vacio."
+
+#: bookwyrm/templates/snippets/shelf.html:67
+msgid "Delete shelf"
+msgstr "Eliminar estante"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:4
+msgid "Change shelf"
+msgstr "Cambiar estante"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:27
+msgid "Unshelve"
+msgstr "Retirar del estante"
+
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:5
+#, python-format
+msgid "Finish \"%(book_title)s \""
+msgstr "Terminar \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_dropdown.html:5
+msgid "More shelves"
+msgstr "Más estantes"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:8
+msgid "Start reading"
+msgstr "Empezar leer"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:13
+msgid "Finish reading"
+msgstr "Terminar de leer"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:16
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
+msgid "Want to read"
+msgstr "Quiero leer"
+
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:5
+#, python-format
+msgid "Start \"%(book_title)s \""
+msgstr "Empezar \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:5
+#, python-format
+msgid "Want to Read \"%(book_title)s \""
+msgstr "Quiero leer \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/status/status.html:9
+msgid "boosted"
+msgstr "respaldó"
+
+#: bookwyrm/templates/snippets/status/status_body.html:24
+#: bookwyrm/templates/snippets/status/status_body.html:37
+#: bookwyrm/templates/snippets/status/status_body.html:38
+msgid "Reply"
+msgstr "Respuesta"
+
+#: bookwyrm/templates/snippets/status/status_content.html:18
+#: bookwyrm/templates/snippets/trimmed_text.html:15
+msgid "Show more"
+msgstr "Mostrar más"
+
+#: bookwyrm/templates/snippets/status/status_content.html:25
+#: bookwyrm/templates/snippets/trimmed_text.html:25
+msgid "Show less"
+msgstr "Mostrar menos"
+
+#: bookwyrm/templates/snippets/status/status_content.html:46
+msgid "Open image in new window"
+msgstr "Abrir imagen en una nueva ventana"
+
+#: bookwyrm/templates/snippets/status/status_header.html:22
+#, fuzzy, python-format
+#| msgid "Added by %(username)s "
+msgid "replied to %(username)s's review "
+msgstr "Agregado por %(username)s "
+
+#: bookwyrm/templates/snippets/status/status_header.html:24
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's comment "
+msgstr "respondió a tu status "
+
+#: bookwyrm/templates/snippets/status/status_header.html:26
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's quote "
+msgstr "respondió a tu status "
+
+#: bookwyrm/templates/snippets/status/status_header.html:28
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's status "
+msgstr "respondió a tu status "
+
+#: bookwyrm/templates/snippets/status/status_options.html:7
+#: bookwyrm/templates/snippets/user_options.html:7
+msgid "More options"
+msgstr "Más opciones"
+
+#: bookwyrm/templates/snippets/status/status_options.html:17
+msgid "Delete status"
+msgstr "Eliminar status"
+
+#: bookwyrm/templates/snippets/status/status_options.html:23
+#: bookwyrm/templates/snippets/user_options.html:13
+msgid "Send direct message"
+msgstr "Enviar mensaje directo"
+
+#: bookwyrm/templates/snippets/switch_edition_button.html:5
+msgid "Switch to this edition"
+msgstr "Cambiar a esta edición"
+
+#: bookwyrm/templates/snippets/tag.html:14
+msgid "Remove tag"
+msgstr "Eliminar etiqueta"
+
+#: bookwyrm/templates/tag.html:9
+#, python-format
+msgid "Books tagged \"%(tag.name)s\""
+msgstr "Libros etiquetados con \"%(tag.name)s\""
+
+#: bookwyrm/templates/user/create_shelf_form.html:5
+#: bookwyrm/templates/user/create_shelf_form.html:22
+msgid "Create Shelf"
+msgstr "Crear estante"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:5
+msgid "Edit Shelf"
+msgstr "Editar estante"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:26
+msgid "Update shelf"
+msgstr "Actualizar estante"
+
+#: bookwyrm/templates/user/followers.html:7
+#: bookwyrm/templates/user/following.html:7 bookwyrm/templates/user/user.html:9
+msgid "User Profile"
+msgstr "Perfil de usuario"
+
+#: bookwyrm/templates/user/followers.html:29
+#, python-format
+msgid "%(username)s has no followers"
+msgstr "%(username)s no tiene seguidores"
+
+#: bookwyrm/templates/user/following.html:13
+msgid "Following"
+msgstr "Siguiendo"
+
+#: bookwyrm/templates/user/following.html:29
+#, python-format
+msgid "%(username)s isn't following any users"
+msgstr "%(username)s no sigue a nadie"
+
+#: bookwyrm/templates/user/lists.html:9
+msgid "Your Lists"
+msgstr "Tus listas"
+
+#: bookwyrm/templates/user/lists.html:11
+#, python-format
+msgid "Lists: %(username)s"
+msgstr "Listas: %(username)s"
+
+#: bookwyrm/templates/user/lists.html:17 bookwyrm/templates/user/lists.html:29
+msgid "Create list"
+msgstr "Crear lista"
+
+#: bookwyrm/templates/user/shelf.html:9
+msgid "Your Shelves"
+msgstr "Tus estantes"
+
+#: bookwyrm/templates/user/shelf.html:11
+#, python-format
+msgid "%(username)s: Shelves"
+msgstr "%(username)s: Estantes"
+
+#: bookwyrm/templates/user/shelf.html:33
+msgid "Create shelf"
+msgstr "Crear estante"
+
+#: bookwyrm/templates/user/shelf.html:54
+msgid "Edit shelf"
+msgstr "Editar estante"
+
+#: bookwyrm/templates/user/user.html:15
+msgid "Edit profile"
+msgstr "Editar perfil"
+
+#: bookwyrm/templates/user/user.html:26
+#: bookwyrm/templates/user/user_layout.html:68
+msgid "Shelves"
+msgstr "Estantes"
+
+#: bookwyrm/templates/user/user.html:31
+#, python-format
+msgid "See all %(size)s"
+msgstr "Ver %(size)s"
+
+#: bookwyrm/templates/user/user.html:44
+#, python-format
+msgid "See all %(shelf_count)s shelves"
+msgstr "Ver los %(shelf_count)s estantes"
+
+#: bookwyrm/templates/user/user.html:56
+#, python-format
+msgid "Set a reading goal for %(year)s"
+msgstr "Establecer una meta de lectura para %(year)s"
+
+#: bookwyrm/templates/user/user.html:62
+msgid "User Activity"
+msgstr "Actividad de usuario"
+
+#: bookwyrm/templates/user/user.html:65
+msgid "RSS feed"
+msgstr "Feed RSS"
+
+#: bookwyrm/templates/user/user.html:76
+msgid "No activities yet!"
+msgstr "¡Aún no actividades!"
+
+#: bookwyrm/templates/user/user_layout.html:32
+msgid "Follow Requests"
+msgstr "Solicitudes de seguidor"
+
+#: bookwyrm/templates/user/user_layout.html:50
+msgid "Activity"
+msgstr "Actividad"
+
+#: bookwyrm/templates/user/user_layout.html:56
+msgid "Reading Goal"
+msgstr "Meta de lectura"
+
+#: bookwyrm/templates/user/user_preview.html:13
+#, python-format
+msgid "Joined %(date)s"
+msgstr "Unido %(date)s"
+
+#: bookwyrm/templates/user/user_preview.html:15
+#, python-format
+msgid "%(counter)s follower"
+msgid_plural "%(counter)s followers"
+msgstr[0] "%(counter)s seguidor"
+msgstr[1] "%(counter)s seguidores"
+
+#: bookwyrm/templates/user/user_preview.html:16
+#, python-format
+msgid "%(counter)s following"
+msgstr "%(counter)s siguiendo"
+
+#~ msgid "Created and curated by"
+#~ msgstr "Creado y comisariado por"
+
+#~ msgid "Created by"
+#~ msgstr "Creado por"
+
+#~ msgid "Added by"
+#~ msgstr "Agregado por"
+
+#~ msgid "Create New Shelf"
+#~ msgstr "Crear nuevo estante"
+
+#~ msgid "Create new list"
+#~ msgstr "Crear nueva lista"
diff --git a/locale/fr_FR/LC_MESSAGES/django.mo b/locale/fr_FR/LC_MESSAGES/django.mo
new file mode 100644
index 00000000..05fdf2c0
Binary files /dev/null and b/locale/fr_FR/LC_MESSAGES/django.mo differ
diff --git a/locale/fr_FR/LC_MESSAGES/django.po b/locale/fr_FR/LC_MESSAGES/django.po
new file mode 100644
index 00000000..38cdd955
--- /dev/null
+++ b/locale/fr_FR/LC_MESSAGES/django.po
@@ -0,0 +1,1934 @@
+# French language text for the bookwyrm UI
+# Copyright (C) 2021 Mouse Reeve
+# This file is distributed under the same license as the bookwyrm package.
+# Mouse Reeve , 2021
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: 0.1.1\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2021-03-07 23:40+0000\n"
+"PO-Revision-Date: 2021-03-02 12:37+0100\n"
+"Last-Translator: Fabien Basmaison \n"
+"Language-Team: Mouse Reeve \n"
+"Language: fr_FR\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=2; plural=(n != 1);\n"
+
+#: bookwyrm/forms.py:185
+msgid "One Day"
+msgstr ""
+
+#: bookwyrm/forms.py:186
+msgid "One Week"
+msgstr ""
+
+#: bookwyrm/forms.py:187
+msgid "One Month"
+msgstr ""
+
+#: bookwyrm/forms.py:188
+msgid "Does Not Expire"
+msgstr ""
+
+#: bookwyrm/forms.py:190
+#, python-format
+msgid "%(count)d uses"
+msgstr ""
+
+#: bookwyrm/forms.py:192
+#, fuzzy
+#| msgid "Unlisted"
+msgid "Unlimited"
+msgstr "Non listé"
+
+#: bookwyrm/models/fields.py:24
+#, python-format
+msgid "%(value)s is not a valid remote_id"
+msgstr ""
+
+#: bookwyrm/models/fields.py:33 bookwyrm/models/fields.py:42
+#, python-format
+msgid "%(value)s is not a valid username"
+msgstr ""
+
+#: bookwyrm/models/fields.py:164 bookwyrm/templates/layout.html:152
+#, fuzzy
+#| msgid "Username:"
+msgid "username"
+msgstr "Nom d’utilisateur :"
+
+#: bookwyrm/models/fields.py:169
+msgid "A user with that username already exists."
+msgstr ""
+
+#: bookwyrm/settings.py:142
+msgid "English"
+msgstr ""
+
+#: bookwyrm/settings.py:143
+msgid "German"
+msgstr ""
+
+#: bookwyrm/settings.py:144
+msgid "Spanish"
+msgstr ""
+
+#: bookwyrm/settings.py:145
+msgid "French"
+msgstr ""
+
+#: bookwyrm/settings.py:146
+msgid "Simplified Chinese"
+msgstr ""
+
+#: bookwyrm/templates/author.html:16 bookwyrm/templates/author.html:17
+#: bookwyrm/templates/edit_author.html:5
+msgid "Edit Author"
+msgstr "Modifier l’auteur ou autrice"
+
+#: bookwyrm/templates/author.html:32
+msgid "Wikipedia"
+msgstr "Wikipedia"
+
+#: bookwyrm/templates/author.html:37
+#, python-format
+msgid "Books by %(name)s"
+msgstr "Livres par %(name)s"
+
+#: bookwyrm/templates/book.html:21
+#: bookwyrm/templates/discover/large-book.html:12
+#: bookwyrm/templates/discover/small-book.html:9
+msgid "by"
+msgstr ""
+
+#: bookwyrm/templates/book.html:29 bookwyrm/templates/book.html:30
+#: bookwyrm/templates/edit_book.html:5
+msgid "Edit Book"
+msgstr "Modifier le livre"
+
+#: bookwyrm/templates/book.html:45
+msgid "Add cover"
+msgstr "Ajouter une couverture"
+
+#: bookwyrm/templates/book.html:51 bookwyrm/templates/lists/list.html:89
+msgid "Add"
+msgstr "Ajouter"
+
+#: bookwyrm/templates/book.html:60
+msgid "ISBN:"
+msgstr "ISBN :"
+
+#: bookwyrm/templates/book.html:67 bookwyrm/templates/edit_book.html:107
+msgid "OCLC Number:"
+msgstr "Numéro OCLC :"
+
+#: bookwyrm/templates/book.html:74 bookwyrm/templates/edit_book.html:111
+msgid "ASIN:"
+msgstr "ASIN :"
+
+#: bookwyrm/templates/book.html:84
+#, fuzzy, python-format
+#| msgid "of %(book.pages)s pages"
+msgid "%(format)s, %(pages)s pages"
+msgstr "sur %(book.pages)s pages"
+
+#: bookwyrm/templates/book.html:86
+#, fuzzy, python-format
+#| msgid "of %(book.pages)s pages"
+msgid "%(pages)s pages"
+msgstr "sur %(book.pages)s pages"
+
+#: bookwyrm/templates/book.html:91
+msgid "View on OpenLibrary"
+msgstr "Voir sur OpenLibrary"
+
+#: bookwyrm/templates/book.html:100
+#, python-format
+msgid "(%(review_count)s review)"
+msgid_plural "(%(review_count)s reviews)"
+msgstr[0] ""
+msgstr[1] ""
+
+#: bookwyrm/templates/book.html:106
+#, fuzzy
+#| msgid "Description:"
+msgid "Add Description"
+msgstr "Ajouter une description"
+
+#: bookwyrm/templates/book.html:113 bookwyrm/templates/edit_book.html:39
+#: bookwyrm/templates/lists/form.html:12
+msgid "Description:"
+msgstr "Description :"
+
+#: bookwyrm/templates/book.html:117 bookwyrm/templates/edit_author.html:78
+#: bookwyrm/templates/edit_book.html:120 bookwyrm/templates/lists/form.html:42
+#: bookwyrm/templates/preferences/edit_user.html:50
+#: bookwyrm/templates/settings/site.html:89
+#: bookwyrm/templates/snippets/progress_update.html:21
+#: bookwyrm/templates/snippets/readthrough.html:64
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:42
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:34
+msgid "Save"
+msgstr "Enregistrer"
+
+#: bookwyrm/templates/book.html:118 bookwyrm/templates/book.html:167
+#: bookwyrm/templates/edit_author.html:79 bookwyrm/templates/edit_book.html:121
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:17
+#: bookwyrm/templates/snippets/goal_form.html:32
+#: bookwyrm/templates/snippets/readthrough.html:65
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:43
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:35
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:28
+msgid "Cancel"
+msgstr "Annuler"
+
+#: bookwyrm/templates/book.html:127
+#, fuzzy, python-format
+#| msgid "Editions of \"%(work_title)s\" "
+msgid "%(count)s editions "
+msgstr "%(title)s par "
+
+#: bookwyrm/templates/book.html:135
+#, fuzzy, python-format
+#| msgid "favorited your %(preview_name)s "
+msgid "This edition is on your %(shelf_name)s shelf."
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/book.html:141
+#, fuzzy, python-format
+#| msgid "replied to your %(preview_name)s "
+msgid "A different edition of this book is on your %(shelf_name)s shelf."
+msgstr " a ajouté %(book_title)s à votre liste « %(list_name)s »"
+
+#: bookwyrm/templates/book.html:150
+msgid "Your reading activity"
+msgstr "Votre activité de lecture"
+
+#: bookwyrm/templates/book.html:152
+#, fuzzy
+#| msgid "Edit read dates"
+msgid "Add read dates"
+msgstr "Ajouter des dates de lecture"
+
+#: bookwyrm/templates/book.html:157
+msgid "You don't have any reading activity for this book."
+msgstr "Vous n’avez aucune activité de lecture pour ce livre"
+
+#: bookwyrm/templates/book.html:164
+msgid "Create"
+msgstr "Créer"
+
+#: bookwyrm/templates/book.html:186
+msgid "Tags"
+msgstr "Tags"
+
+#: bookwyrm/templates/book.html:190 bookwyrm/templates/snippets/tag.html:18
+msgid "Add tag"
+msgstr "Ajouter un tag"
+
+#: bookwyrm/templates/book.html:207
+msgid "Subjects"
+msgstr "Sujets"
+
+#: bookwyrm/templates/book.html:218
+msgid "Places"
+msgstr "Lieux"
+
+#: bookwyrm/templates/book.html:229 bookwyrm/templates/layout.html:64
+#: bookwyrm/templates/lists/lists.html:4 bookwyrm/templates/lists/lists.html:9
+#: bookwyrm/templates/search_results.html:92
+#: bookwyrm/templates/user/user_layout.html:62
+msgid "Lists"
+msgstr "Listes"
+
+#: bookwyrm/templates/book.html:258
+msgid "rated it"
+msgstr "l’a noté"
+
+#: bookwyrm/templates/components/inline_form.html:8
+#: bookwyrm/templates/feed/feed_layout.html:54
+#, fuzzy
+#| msgid "Closed"
+msgid "Close"
+msgstr "Fermer"
+
+#: bookwyrm/templates/discover/about.html:7
+#, fuzzy, python-format
+#| msgid "Join %(name)s"
+msgid "About %(site_name)s"
+msgstr "À propos de %(name)s"
+
+#: bookwyrm/templates/discover/about.html:10
+#: bookwyrm/templates/discover/about.html:20
+msgid "Code of Conduct"
+msgstr "Code de conduite"
+
+#: bookwyrm/templates/discover/about.html:13
+#: bookwyrm/templates/discover/about.html:29
+msgid "Privacy Policy"
+msgstr "Politique de vie privée"
+
+#: bookwyrm/templates/discover/discover.html:6
+msgid "Recent Books"
+msgstr "Livres récents"
+
+#: bookwyrm/templates/discover/landing_layout.html:5
+msgid "Welcome"
+msgstr "Bienvenue"
+
+#: bookwyrm/templates/discover/landing_layout.html:17
+msgid "Decentralized"
+msgstr "Décentralisé"
+
+#: bookwyrm/templates/discover/landing_layout.html:23
+msgid "Friendly"
+msgstr "Sympa"
+
+#: bookwyrm/templates/discover/landing_layout.html:29
+msgid "Anti-Corporate"
+msgstr "Anti‑commercial"
+
+#: bookwyrm/templates/discover/landing_layout.html:44
+#, python-format
+msgid "Join %(name)s"
+msgstr "Rejoignez %(name)s"
+
+#: bookwyrm/templates/discover/landing_layout.html:49
+#: bookwyrm/templates/login.html:48
+msgid "This instance is closed"
+msgstr "Cette instance est fermée"
+
+#: bookwyrm/templates/discover/landing_layout.html:55
+msgid "Your Account"
+msgstr "Votre compte"
+
+#: bookwyrm/templates/edit_author.html:13 bookwyrm/templates/edit_book.html:13
+msgid "Added:"
+msgstr "AJouté :"
+
+#: bookwyrm/templates/edit_author.html:14 bookwyrm/templates/edit_book.html:14
+msgid "Updated:"
+msgstr "Mis à jour :"
+
+#: bookwyrm/templates/edit_author.html:15 bookwyrm/templates/edit_book.html:15
+msgid "Last edited by:"
+msgstr "Dernière modification par :"
+
+#: bookwyrm/templates/edit_author.html:31 bookwyrm/templates/edit_book.html:30
+msgid "Metadata"
+msgstr "Métadonnées"
+
+#: bookwyrm/templates/edit_author.html:32 bookwyrm/templates/lists/form.html:8
+#: bookwyrm/templates/user/create_shelf_form.html:13
+#: bookwyrm/templates/user/edit_shelf_form.html:14
+msgid "Name:"
+msgstr "Nom :"
+
+#: bookwyrm/templates/edit_author.html:37
+msgid "Bio:"
+msgstr "Bio :"
+
+#: bookwyrm/templates/edit_author.html:42
+msgid "Wikipedia link:"
+msgstr "Wikipedia :"
+
+#: bookwyrm/templates/edit_author.html:47
+msgid "Birth date:"
+msgstr "Date de naissance :"
+
+#: bookwyrm/templates/edit_author.html:52
+msgid "Death date:"
+msgstr "Date de décès :"
+
+#: bookwyrm/templates/edit_author.html:58
+msgid "Author Identifiers"
+msgstr "Identifiants de l’auteur ou autrice"
+
+#: bookwyrm/templates/edit_author.html:59 bookwyrm/templates/edit_book.html:103
+msgid "Openlibrary key:"
+msgstr "Clé Openlibrary :"
+
+#: bookwyrm/templates/edit_author.html:64
+msgid "Librarything key:"
+msgstr "Clé Librarything :"
+
+#: bookwyrm/templates/edit_author.html:69
+msgid "Goodreads key:"
+msgstr "Clé Goodreads :"
+
+#: bookwyrm/templates/edit_book.html:31
+msgid "Title:"
+msgstr "Titre :"
+
+#: bookwyrm/templates/edit_book.html:35
+msgid "Subtitle:"
+msgstr "Sous‑titre :"
+
+#: bookwyrm/templates/edit_book.html:43
+msgid "Series:"
+msgstr "Série :"
+
+#: bookwyrm/templates/edit_book.html:47
+msgid "Series number:"
+msgstr "Numéro dans la série :"
+
+#: bookwyrm/templates/edit_book.html:51
+msgid "First published date:"
+msgstr "Première date de publication :"
+
+#: bookwyrm/templates/edit_book.html:55
+msgid "Published date:"
+msgstr "Date de publication :"
+
+#: bookwyrm/templates/edit_book.html:68
+#: bookwyrm/templates/snippets/shelf.html:9
+msgid "Cover"
+msgstr "Couverture"
+
+#: bookwyrm/templates/edit_book.html:78
+msgid "Physical Properties"
+msgstr "Propriétés physiques"
+
+#: bookwyrm/templates/edit_book.html:79
+msgid "Format:"
+msgstr "Format :"
+
+#: bookwyrm/templates/edit_book.html:87
+msgid "Pages:"
+msgstr "Pages :"
+
+#: bookwyrm/templates/edit_book.html:94
+msgid "Book Identifiers"
+msgstr "Identifiants du livre"
+
+#: bookwyrm/templates/edit_book.html:95
+msgid "ISBN 13:"
+msgstr "ISBN 13 :"
+
+#: bookwyrm/templates/edit_book.html:99
+msgid "ISBN 10:"
+msgstr "ISBN 10 :"
+
+#: bookwyrm/templates/editions.html:5
+#, fuzzy, python-format
+#| msgid "Finish \"%(book_title)s \""
+msgid "Editions of %(book_title)s"
+msgstr "Éditions de %(book_title)s"
+
+#: bookwyrm/templates/editions.html:9
+#, python-format
+msgid "Editions of \"%(work_title)s\" "
+msgstr "Éditions de « %(work_title)s » "
+
+#: bookwyrm/templates/error.html:4
+msgid "Oops!"
+msgstr "Oups !"
+
+#: bookwyrm/templates/error.html:8
+msgid "Server Error"
+msgstr "Erreur côté serveur"
+
+#: bookwyrm/templates/error.html:9
+msgid "Something went wrong! Sorry about that."
+msgstr "Une erreur s’est produite ; désolé !"
+
+#: bookwyrm/templates/feed/direct_messages.html:8
+#, python-format
+msgid "Direct Messages with %(username)s "
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/feed/direct_messages.html:10
+#: bookwyrm/templates/layout.html:87
+msgid "Direct Messages"
+msgstr "Messages directs"
+
+#: bookwyrm/templates/feed/direct_messages.html:13
+msgid "All messages"
+msgstr "Tous les messages"
+
+#: bookwyrm/templates/feed/direct_messages.html:22
+msgid "You have no messages right now."
+msgstr "Vous n’avez aucun message pour l’instant."
+
+#: bookwyrm/templates/feed/feed.html:6
+#, python-format
+msgid "%(tab_title)s Timeline"
+msgstr "%(tab_title)s — Fil d’actualité"
+
+#: bookwyrm/templates/feed/feed.html:10 bookwyrm/views/feed.py:33
+msgid "Home"
+msgstr "Accueil"
+
+#: bookwyrm/templates/feed/feed.html:13 bookwyrm/views/feed.py:37
+msgid "Local"
+msgstr "Local"
+
+#: bookwyrm/templates/feed/feed.html:16 bookwyrm/views/feed.py:41
+msgid "Federated"
+msgstr "Fédéré"
+
+#: bookwyrm/templates/feed/feed.html:24
+msgid "Announcements"
+msgstr "Annonces"
+
+#: bookwyrm/templates/feed/feed.html:32
+msgid "There aren't any activities right now! Try following a user to get started"
+msgstr "Aucune activité pour l’instant ! Abonnez‑vous à quelqu’un pour commencer"
+
+#: bookwyrm/templates/feed/feed_layout.html:5
+#, fuzzy
+#| msgid "Updated:"
+msgid "Updates"
+msgstr "Mises à jour"
+
+#: bookwyrm/templates/feed/feed_layout.html:11
+msgid "Your books"
+msgstr "Vos livres"
+
+#: bookwyrm/templates/feed/feed_layout.html:13
+msgid "There are no books here right now! Try searching for a book to get started"
+msgstr "Aucun livre ici pour l’instant ! Cherchez un livre pour commencer"
+
+#: bookwyrm/templates/feed/feed_layout.html:23
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Read"
+msgid "To Read"
+msgstr "Lu"
+
+#: bookwyrm/templates/feed/feed_layout.html:24
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Started reading"
+msgid "Currently Reading"
+msgstr "Commencer la lecture"
+
+#: bookwyrm/templates/feed/feed_layout.html:25
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:11
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Read"
+msgstr "Lu"
+
+#: bookwyrm/templates/feed/feed_layout.html:76 bookwyrm/templates/goal.html:26
+#: bookwyrm/templates/snippets/goal_card.html:6
+#, python-format
+msgid "%(year)s Reading Goal"
+msgstr "Défi lecture pour %(year)s"
+
+#: bookwyrm/templates/feed/status.html:8
+msgid "Back"
+msgstr "Retour"
+
+#: bookwyrm/templates/goal.html:7
+#, python-format
+msgid "%(year)s Reading Progress"
+msgstr "Progression de lecture pour %(year)s"
+
+#: bookwyrm/templates/goal.html:11
+#, fuzzy
+#| msgid "Edit Book"
+msgid "Edit Goal"
+msgstr "Modifier le défi"
+
+#: bookwyrm/templates/goal.html:30
+#: bookwyrm/templates/snippets/goal_card.html:13
+#, python-format
+msgid "Set a goal for how many books you'll finish reading in %(year)s, and track your progress throughout the year."
+msgstr "Définissez un nombre de livre à lire comme objectif pour %(year)s, et suivezvotre progression au fil de l’année."
+
+#: bookwyrm/templates/goal.html:39
+#, python-format
+msgid "%(name)s hasn't set a reading goal for %(year)s."
+msgstr "%(name)s n’a aucun défi lecture pour %(year)s."
+
+#: bookwyrm/templates/goal.html:51
+#, fuzzy, python-format
+#| msgid "Your books"
+msgid "Your %(year)s Books"
+msgstr "Vos livres en %(year)s"
+
+#: bookwyrm/templates/goal.html:53
+#, fuzzy, python-format
+#| msgid "%(username)s has no followers"
+msgid "%(username)s's %(year)s Books"
+msgstr "Livres de %(username)s en %(year)s"
+
+#: bookwyrm/templates/import.html:5 bookwyrm/templates/import.html:9
+#: bookwyrm/templates/layout.html:102
+msgid "Import Books"
+msgstr "Importer des livres"
+
+#: bookwyrm/templates/import.html:14
+msgid "Data source"
+msgstr "Source de données"
+
+#: bookwyrm/templates/import.html:32
+msgid "Include reviews"
+msgstr "Importer les critiques"
+
+#: bookwyrm/templates/import.html:37
+msgid "Privacy setting for imported reviews:"
+msgstr "Confidentialité des critiques importées :"
+
+#: bookwyrm/templates/import.html:41
+msgid "Import"
+msgstr "Importer"
+
+#: bookwyrm/templates/import.html:46
+msgid "Recent Imports"
+msgstr "Importations récentes"
+
+#: bookwyrm/templates/import.html:48
+msgid "No recent imports"
+msgstr "Aucune importation récente"
+
+#: bookwyrm/templates/import_status.html:6
+#: bookwyrm/templates/import_status.html:10
+msgid "Import Status"
+msgstr "Statut de l’importation"
+
+#: bookwyrm/templates/import_status.html:13
+msgid "Import started:"
+msgstr "Importation en cours :"
+
+#: bookwyrm/templates/import_status.html:17
+msgid "Import completed:"
+msgstr "Importation terminé :"
+
+#: bookwyrm/templates/import_status.html:20
+msgid "TASK FAILED"
+msgstr "la tâche a échoué"
+
+#: bookwyrm/templates/import_status.html:26
+msgid "Import still in progress."
+msgstr "L’importation est toujours en cours"
+
+#: bookwyrm/templates/import_status.html:28
+msgid "(Hit reload to update!)"
+msgstr "(Rechargez la page pour mettre à jour !"
+
+#: bookwyrm/templates/import_status.html:35
+msgid "Failed to load"
+msgstr "Items non importés"
+
+#: bookwyrm/templates/import_status.html:44
+#, python-format
+msgid "Jump to the bottom of the list to select the %(failed_count)s items which failed to import."
+msgstr "Sauter en bas de liste pour sélectionner les %(failed_count)s items n’ayant pu être importés."
+
+#: bookwyrm/templates/import_status.html:79
+msgid "Select all"
+msgstr "Tout sélectionner"
+
+#: bookwyrm/templates/import_status.html:82
+msgid "Retry items"
+msgstr "Essayer d’importer les items sélectionnés de nouveau"
+
+#: bookwyrm/templates/import_status.html:108
+msgid "Successfully imported"
+msgstr "Importation réussie"
+
+#: bookwyrm/templates/import_status.html:112
+#: bookwyrm/templates/lists/curate.html:14
+msgid "Book"
+msgstr "Livre"
+
+#: bookwyrm/templates/import_status.html:115
+#: bookwyrm/templates/snippets/create_status_form.html:10
+#: bookwyrm/templates/snippets/shelf.html:10
+msgid "Title"
+msgstr "Titre"
+
+#: bookwyrm/templates/import_status.html:118
+#: bookwyrm/templates/snippets/shelf.html:11
+msgid "Author"
+msgstr "Auteur ou autrice"
+
+#: bookwyrm/templates/import_status.html:141
+msgid "Imported"
+msgstr "Importé"
+
+#: bookwyrm/templates/invite.html:4 bookwyrm/templates/invite.html:12
+#: bookwyrm/templates/login.html:43
+msgid "Create an Account"
+msgstr "Créer un compte"
+
+#: bookwyrm/templates/invite.html:21
+msgid "Permission Denied"
+msgstr "Autorisation refusée"
+
+#: bookwyrm/templates/invite.html:22
+msgid "Sorry! This invite code is no longer valid."
+msgstr "Cette invitation n’est plus valide ; désolé !"
+
+#: bookwyrm/templates/isbn_search_results.html:4
+#: bookwyrm/templates/search_results.html:4
+msgid "Search Results"
+msgstr "Résultats de recherche"
+
+#: bookwyrm/templates/isbn_search_results.html:9
+#: bookwyrm/templates/search_results.html:9
+#, python-format
+msgid "Search Results for \"%(query)s\""
+msgstr "Résultats de recherche pour « %(query)s »"
+
+#: bookwyrm/templates/isbn_search_results.html:14
+#: bookwyrm/templates/search_results.html:14
+msgid "Matching Books"
+msgstr "Livres correspondants"
+
+#: bookwyrm/templates/isbn_search_results.html:17
+#: bookwyrm/templates/search_results.html:17
+#, python-format
+msgid "No books found for \"%(query)s\""
+msgstr "Aucun livre trouvé pour « %(query)s »"
+
+#: bookwyrm/templates/layout.html:33
+msgid "Search for a book or user"
+msgstr "Chercher un livre ou un compte"
+
+#: bookwyrm/templates/layout.html:37 bookwyrm/templates/layout.html:38
+#: bookwyrm/templates/lists/list.html:62
+msgid "Search"
+msgstr "Chercher"
+
+#: bookwyrm/templates/layout.html:47 bookwyrm/templates/layout.html:48
+msgid "Main navigation menu"
+msgstr "Menu de navigation principal "
+
+#: bookwyrm/templates/layout.html:58
+msgid "Your shelves"
+msgstr "Vos étagères"
+
+#: bookwyrm/templates/layout.html:61
+msgid "Feed"
+msgstr "Fil d’actualité"
+
+#: bookwyrm/templates/layout.html:92
+#: bookwyrm/templates/preferences/preferences_layout.html:14
+msgid "Profile"
+msgstr "Profil"
+
+#: bookwyrm/templates/layout.html:97
+#, fuzzy
+#| msgid "Instance Settings"
+msgid "Settings"
+msgstr "Paramètres de l’instance"
+
+#: bookwyrm/templates/layout.html:111
+#: bookwyrm/templates/settings/admin_layout.html:19
+#: bookwyrm/templates/settings/manage_invites.html:3
+msgid "Invites"
+msgstr "Invitations"
+
+#: bookwyrm/templates/layout.html:118
+msgid "Site Configuration"
+msgstr "Configuration du site"
+
+#: bookwyrm/templates/layout.html:125
+msgid "Log out"
+msgstr "Se déconnecter"
+
+#: bookwyrm/templates/layout.html:133 bookwyrm/templates/layout.html:134
+#: bookwyrm/templates/notifications.html:6
+#: bookwyrm/templates/notifications.html:10
+msgid "Notifications"
+msgstr "Notifications"
+
+#: bookwyrm/templates/layout.html:151 bookwyrm/templates/layout.html:155
+#: bookwyrm/templates/login.html:17
+#: bookwyrm/templates/snippets/register_form.html:4
+msgid "Username:"
+msgstr "Nom d’utilisateur :"
+
+#: bookwyrm/templates/layout.html:156
+#, fuzzy
+#| msgid "Password:"
+msgid "password"
+msgstr "Mot de passe :"
+
+#: bookwyrm/templates/layout.html:157 bookwyrm/templates/login.html:36
+msgid "Forgot your password?"
+msgstr "Mot de passe oublié ?"
+
+#: bookwyrm/templates/layout.html:160 bookwyrm/templates/login.html:10
+#: bookwyrm/templates/login.html:33
+msgid "Log in"
+msgstr "Se connecter"
+
+#: bookwyrm/templates/layout.html:168
+msgid "Join"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:191
+msgid "About this server"
+msgstr "À propos de ce serveur"
+
+#: bookwyrm/templates/layout.html:195
+msgid "Contact site admin"
+msgstr "Contacter l’administrateur du site"
+
+#: bookwyrm/templates/layout.html:202
+#, python-format
+msgid "Support %(site_name)s on %(support_title)s "
+msgstr ""
+
+#: bookwyrm/templates/layout.html:206
+msgid "BookWyrm is open source software. You can contribute or report issues on GitHub ."
+msgstr "Bookwyrm est un logiciel libre. Vous pouvez contribuer ou faire des rapports de bogues via GitHub ."
+
+#: bookwyrm/templates/lists/create_form.html:5
+#: bookwyrm/templates/lists/lists.html:17
+msgid "Create List"
+msgstr "Créer une liste"
+
+#: bookwyrm/templates/lists/created_text.html:5
+#, fuzzy, python-format
+#| msgid "favorited your %(preview_name)s "
+msgid "Created and curated by %(username)s "
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/lists/created_text.html:7
+#, fuzzy, python-format
+#| msgid "favorited your %(preview_name)s "
+msgid "Created by %(username)s "
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/lists/curate.html:6
+msgid "Pending Books"
+msgstr "Livres en attente de modération"
+
+#: bookwyrm/templates/lists/curate.html:7
+msgid "Go to list"
+msgstr "Aller à la liste"
+
+#: bookwyrm/templates/lists/curate.html:9
+msgid "You're all set!"
+msgstr "Aucun livre en attente de validation !"
+
+#: bookwyrm/templates/lists/curate.html:15
+msgid "Suggested by"
+msgstr "Suggéré par"
+
+#: bookwyrm/templates/lists/curate.html:35
+msgid "Approve"
+msgstr "Approuver"
+
+#: bookwyrm/templates/lists/curate.html:41
+msgid "Discard"
+msgstr "Rejeter"
+
+#: bookwyrm/templates/lists/edit_form.html:5
+#: bookwyrm/templates/lists/list_layout.html:18
+msgid "Edit List"
+msgstr "Modifier la liste"
+
+#: bookwyrm/templates/lists/form.html:18
+msgid "List curation:"
+msgstr "Modération de la liste :"
+
+#: bookwyrm/templates/lists/form.html:21
+msgid "Closed"
+msgstr "Fermée"
+
+#: bookwyrm/templates/lists/form.html:22
+msgid "Only you can add and remove books to this list"
+msgstr "Vous seul(e) pouvez ajouter ou supprimer des livres dans cette liste"
+
+#: bookwyrm/templates/lists/form.html:26
+msgid "Curated"
+msgstr "Modérée"
+
+#: bookwyrm/templates/lists/form.html:27
+msgid "Anyone can suggest books, subject to your approval"
+msgstr "N’importe qui peut suggérer des livres, soumis à votre approbation"
+
+#: bookwyrm/templates/lists/form.html:31
+msgid "Open"
+msgstr "Ouverte"
+
+#: bookwyrm/templates/lists/form.html:32
+msgid "Anyone can add books to this list"
+msgstr "N’importe qui peut suggérer des livres"
+
+#: bookwyrm/templates/lists/list.html:17
+msgid "This list is currently empty"
+msgstr "Cette liste est vide actuellement"
+
+#: bookwyrm/templates/lists/list.html:35
+#, fuzzy, python-format
+#| msgid "favorited your %(preview_name)s "
+msgid "Added by %(username)s "
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/lists/list.html:41
+msgid "Remove"
+msgstr "Supprimer"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Add Books"
+msgstr "Ajouter des livres"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Suggest Books"
+msgstr "Suggérer des livres"
+
+#: bookwyrm/templates/lists/list.html:58
+msgid "Search for a book"
+msgstr "Chercher un livre"
+
+#: bookwyrm/templates/lists/list.html:63
+msgid "search"
+msgstr "Chercher"
+
+#: bookwyrm/templates/lists/list.html:69
+msgid "Clear search"
+msgstr "Vider la requête"
+
+#: bookwyrm/templates/lists/list.html:74
+#, python-format
+msgid "No books found matching the query \"%(query)s\""
+msgstr "Aucun livre trouvé pour la requête « %(query)s »"
+
+#: bookwyrm/templates/lists/list.html:75
+msgid "No books found"
+msgstr "Aucun livre trouvé"
+
+#: bookwyrm/templates/lists/list.html:89
+msgid "Suggest"
+msgstr "Suggérer"
+
+#: bookwyrm/templates/lists/lists.html:14
+msgid "Your lists"
+msgstr "Vos listes"
+
+#: bookwyrm/templates/lists/lists.html:32
+#, fuzzy, python-format
+#| msgid "See all %(size)s"
+msgid "See all %(size)s lists"
+msgstr "Voir les %(size)s"
+
+#: bookwyrm/templates/lists/lists.html:40
+msgid "Recent Lists"
+msgstr "Listes récentes"
+
+#: bookwyrm/templates/login.html:4
+msgid "Login"
+msgstr "Connexion"
+
+#: bookwyrm/templates/login.html:23 bookwyrm/templates/password_reset.html:17
+#: bookwyrm/templates/snippets/register_form.html:22
+msgid "Password:"
+msgstr "Mot de passe :"
+
+#: bookwyrm/templates/login.html:49
+msgid "Contact an administrator to get an invite"
+msgstr "Contacter un administrateur pour obtenir une invitation"
+
+#: bookwyrm/templates/login.html:59
+msgid "More about this site"
+msgstr "En savoir plus sur ce site"
+
+#: bookwyrm/templates/notfound.html:4 bookwyrm/templates/notfound.html:8
+msgid "Not Found"
+msgstr "Introuvable"
+
+#: bookwyrm/templates/notfound.html:9
+msgid "The page you requested doesn't seem to exist!"
+msgstr "Il semblerait que la page que vous avez demandée n’existe pas !"
+
+#: bookwyrm/templates/notifications.html:14
+msgid "Delete notifications"
+msgstr "Supprimer les notifications"
+
+#: bookwyrm/templates/notifications.html:51
+#, python-format
+msgid "favorited your review of %(book_title)s "
+msgstr "a ajouté votre critique de %(book_title)s à ses favoris"
+
+#: bookwyrm/templates/notifications.html:53
+#, python-format
+msgid "favorited your comment on %(book_title)s "
+msgstr "a ajouté votre commentaire sur %(book_title)s à ses favoris"
+
+#: bookwyrm/templates/notifications.html:55
+#, python-format
+msgid "favorited your quote from %(book_title)s "
+msgstr "a ajouté votre citation de %(book_title)s à ses favoris"
+
+#: bookwyrm/templates/notifications.html:57
+#, python-format
+msgid "favorited your status "
+msgstr "a ajouté votre statut à ses favoris"
+
+#: bookwyrm/templates/notifications.html:62
+#, python-format
+msgid "mentioned you in a review of %(book_title)s "
+msgstr "vous a mentionné dans sa critique de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:64
+#, python-format
+msgid "mentioned you in a comment on %(book_title)s "
+msgstr "vous a mentionné dans son commentaire sur %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:66
+#, python-format
+msgid "mentioned you in a quote from %(book_title)s "
+msgstr "vous a mentionné dans sa citation de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:68
+#, python-format
+msgid "mentioned you in a status "
+msgstr "vous a mentionné dans son statut "
+
+#: bookwyrm/templates/notifications.html:73
+#, python-format
+msgid "replied to your review of %(book_title)s "
+msgstr "a répondu à votre critique de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:75
+#, python-format
+msgid "replied to your comment on %(book_title)s "
+msgstr "a répondu à votre commentaire sur %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:77
+#, python-format
+msgid "replied to your quote from %(book_title)s "
+msgstr "a répondu à votre citation de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:79
+#, python-format
+msgid "replied to your status "
+msgstr "a répondu à votre statut "
+
+#: bookwyrm/templates/notifications.html:83
+msgid "followed you"
+msgstr "s’est abonné(e)"
+
+#: bookwyrm/templates/notifications.html:86
+msgid "sent you a follow request"
+msgstr "vous a envoyé une demande d’abonnement"
+
+#: bookwyrm/templates/notifications.html:92
+#, python-format
+msgid "boosted your review of %(book_title)s "
+msgstr "a partagé votre critique de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:94
+#, python-format
+msgid "boosted your comment on%(book_title)s "
+msgstr "a partagé votre commentaire sur %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:96
+#, python-format
+msgid "boosted your quote from %(book_title)s "
+msgstr "a partagé votre citation de %(book_title)s "
+
+#: bookwyrm/templates/notifications.html:98
+#, python-format
+msgid "boosted your status "
+msgstr "a partagé votre statut "
+
+#: bookwyrm/templates/notifications.html:102
+#, python-format
+msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgstr " a ajouté %(book_title)s à votre liste « %(list_name)s »"
+
+#: bookwyrm/templates/notifications.html:104
+#, python-format
+msgid " suggested adding %(book_title)s to your list \"%(list_name)s \""
+msgstr " a suggégré l’ajout de %(book_title)s à votre liste « %(list_name)s »"
+
+#: bookwyrm/templates/notifications.html:108
+#, python-format
+msgid " your import completed."
+msgstr " votre importation est terminée."
+
+#: bookwyrm/templates/notifications.html:142
+msgid "You're all caught up!"
+msgstr "Aucune nouvelle notification !"
+
+#: bookwyrm/templates/password_reset.html:4
+#: bookwyrm/templates/password_reset.html:10
+#: bookwyrm/templates/password_reset_request.html:4
+#: bookwyrm/templates/password_reset_request.html:10
+msgid "Reset Password"
+msgstr "Changez le mot de passe"
+
+#: bookwyrm/templates/password_reset.html:23
+#: bookwyrm/templates/preferences/change_password.html:18
+msgid "Confirm password:"
+msgstr "Confirmez le mot de passe :"
+
+#: bookwyrm/templates/password_reset.html:30
+msgid "Confirm"
+msgstr "Confirmer"
+
+#: bookwyrm/templates/password_reset_request.html:12
+msgid "A link to reset your password will be sent to your email address"
+msgstr "Un lien pour changer votre mot de passe sera envoyé à votre addresse email"
+
+#: bookwyrm/templates/password_reset_request.html:16
+#: bookwyrm/templates/preferences/edit_user.html:38
+#: bookwyrm/templates/snippets/register_form.html:13
+msgid "Email address:"
+msgstr "Adresse email :"
+
+#: bookwyrm/templates/password_reset_request.html:23
+msgid "Reset password"
+msgstr "Changer de mot de passe"
+
+#: bookwyrm/templates/preferences/blocks.html:4
+#: bookwyrm/templates/preferences/blocks.html:7
+#: bookwyrm/templates/preferences/preferences_layout.html:23
+msgid "Blocked Users"
+msgstr "Comptes bloqués"
+
+#: bookwyrm/templates/preferences/blocks.html:12
+msgid "No users currently blocked."
+msgstr "Aucun compte bloqué actuellement"
+
+#: bookwyrm/templates/preferences/change_password.html:4
+#: bookwyrm/templates/preferences/change_password.html:7
+#: bookwyrm/templates/preferences/change_password.html:21
+#: bookwyrm/templates/preferences/preferences_layout.html:17
+msgid "Change Password"
+msgstr "Changer le mot de passe"
+
+#: bookwyrm/templates/preferences/change_password.html:14
+msgid "New password:"
+msgstr "Nouveau mot de passe :"
+
+#: bookwyrm/templates/preferences/edit_user.html:4
+#: bookwyrm/templates/preferences/edit_user.html:7
+msgid "Edit Profile"
+msgstr "Modifier le profil"
+
+#: bookwyrm/templates/preferences/edit_user.html:17
+msgid "Avatar:"
+msgstr "Avatar :"
+
+#: bookwyrm/templates/preferences/edit_user.html:24
+msgid "Display name:"
+msgstr "Nom affiché :"
+
+#: bookwyrm/templates/preferences/edit_user.html:31
+msgid "Summary:"
+msgstr "Résumé :"
+
+#: bookwyrm/templates/preferences/edit_user.html:46
+msgid "Manually approve followers:"
+msgstr "Autoriser les abonnements manuellement :"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:11
+msgid "Account"
+msgstr "Compte"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:20
+msgid "Relationships"
+msgstr "Relations"
+
+#: bookwyrm/templates/search_results.html:33
+msgid "Didn't find what you were looking for?"
+msgstr "Vous n’avez pas trouvé ce que vous cherchiez ?"
+
+#: bookwyrm/templates/search_results.html:35
+msgid "Show results from other catalogues"
+msgstr "Montrer les résultats d’autres catalogues"
+
+#: bookwyrm/templates/search_results.html:57
+msgid "Import book"
+msgstr "Importer le livre"
+
+#: bookwyrm/templates/search_results.html:67
+msgid "Hide results from other catalogues"
+msgstr "Masquer les résultats d’autres catalogues"
+
+#: bookwyrm/templates/search_results.html:75
+msgid "Matching Users"
+msgstr "Comptes correspondants"
+
+#: bookwyrm/templates/search_results.html:77
+#, python-format
+msgid "No users found for \"%(query)s\""
+msgstr "Aucun compte trouvé pour « %(query)s »"
+
+#: bookwyrm/templates/search_results.html:94
+#, python-format
+msgid "No lists found for \"%(query)s\""
+msgstr "Aucune liste trouvée pour « %(query)s »"
+
+#: bookwyrm/templates/settings/admin_layout.html:4
+msgid "Administration"
+msgstr "Administration"
+
+#: bookwyrm/templates/settings/admin_layout.html:15
+msgid "Manage Users"
+msgstr "Gérer les comptes"
+
+#: bookwyrm/templates/settings/admin_layout.html:23
+#: bookwyrm/templates/settings/federation.html:4
+msgid "Federated Servers"
+msgstr "Serveurs fédérés"
+
+#: bookwyrm/templates/settings/admin_layout.html:28
+msgid "Instance Settings"
+msgstr "Paramètres de l’instance"
+
+#: bookwyrm/templates/settings/admin_layout.html:32
+#: bookwyrm/templates/settings/site.html:4
+#: bookwyrm/templates/settings/site.html:6
+#, fuzzy
+#| msgid "Instance Settings"
+msgid "Site Settings"
+msgstr "Paramètres du site"
+
+#: bookwyrm/templates/settings/admin_layout.html:35
+#: bookwyrm/templates/settings/site.html:13
+msgid "Instance Info"
+msgstr "Information sur l’instance"
+
+#: bookwyrm/templates/settings/admin_layout.html:36
+#: bookwyrm/templates/settings/site.html:39
+msgid "Images"
+msgstr "Images"
+
+#: bookwyrm/templates/settings/admin_layout.html:37
+#: bookwyrm/templates/settings/site.html:59
+msgid "Footer Content"
+msgstr "Contenu du pied de page"
+
+#: bookwyrm/templates/settings/admin_layout.html:38
+#: bookwyrm/templates/settings/site.html:77
+msgid "Registration"
+msgstr "Enregistrement"
+
+#: bookwyrm/templates/settings/federation.html:10
+msgid "Server name"
+msgstr "Nom du serveur"
+
+#: bookwyrm/templates/settings/federation.html:11
+msgid "Software"
+msgstr "Logiciel"
+
+#: bookwyrm/templates/settings/federation.html:12
+msgid "Status"
+msgstr "Statut"
+
+#: bookwyrm/templates/settings/manage_invites.html:7
+msgid "Generate New Invite"
+msgstr "Générer une nouvelle invitation"
+
+#: bookwyrm/templates/settings/manage_invites.html:13
+msgid "Expiry:"
+msgstr "Expiration :"
+
+#: bookwyrm/templates/settings/manage_invites.html:19
+msgid "Use limit:"
+msgstr "Limiter à :"
+
+#: bookwyrm/templates/settings/manage_invites.html:26
+msgid "Create Invite"
+msgstr "Créer une invitation"
+
+#: bookwyrm/templates/settings/manage_invites.html:33
+msgid "Link"
+msgstr "Lien"
+
+#: bookwyrm/templates/settings/manage_invites.html:34
+msgid "Expires"
+msgstr "Expiration"
+
+#: bookwyrm/templates/settings/manage_invites.html:35
+msgid "Max uses"
+msgstr "Nombre maximum d’utilisations"
+
+#: bookwyrm/templates/settings/manage_invites.html:36
+msgid "Times used"
+msgstr "Nombre de fois utilisée"
+
+#: bookwyrm/templates/settings/manage_invites.html:39
+msgid "No active invites"
+msgstr "Aucune invitation active"
+
+#: bookwyrm/templates/settings/site.html:15
+msgid "Instance Name:"
+msgstr "Nom de l’instance :"
+
+#: bookwyrm/templates/settings/site.html:19
+msgid "Tagline:"
+msgstr "Slogan :"
+
+#: bookwyrm/templates/settings/site.html:23
+msgid "Instance description:"
+msgstr "Description de l’instance :"
+
+#: bookwyrm/templates/settings/site.html:27
+msgid "Code of conduct:"
+msgstr "Code de conduite :"
+
+#: bookwyrm/templates/settings/site.html:31
+msgid "Privacy Policy:"
+msgstr "Politique de vie privée :"
+
+#: bookwyrm/templates/settings/site.html:42
+msgid "Logo:"
+msgstr "Logo :"
+
+#: bookwyrm/templates/settings/site.html:46
+msgid "Logo small:"
+msgstr "Logo réduit :"
+
+#: bookwyrm/templates/settings/site.html:50
+msgid "Favicon:"
+msgstr "Favicon :"
+
+#: bookwyrm/templates/settings/site.html:61
+msgid "Support link:"
+msgstr "URL pour soutenir l’instance :"
+
+#: bookwyrm/templates/settings/site.html:65
+msgid "Support title:"
+msgstr "Titre pour soutenir l’instance :"
+
+#: bookwyrm/templates/settings/site.html:69
+msgid "Admin email:"
+msgstr "Email de l’administrateur :"
+
+#: bookwyrm/templates/settings/site.html:79
+msgid "Allow registration:"
+msgstr "Autoriser l’enregistrement :"
+
+#: bookwyrm/templates/settings/site.html:83
+msgid "Registration closed text:"
+msgstr "Texte affiché lorsque les enregistrements sont clos :"
+
+#: bookwyrm/templates/snippets/block_button.html:5
+msgid "Block"
+msgstr "Bloquer"
+
+#: bookwyrm/templates/snippets/block_button.html:10
+msgid "Un-block"
+msgstr "Débloquer"
+
+#: bookwyrm/templates/snippets/book_titleby.html:3
+#, python-format
+msgid "%(title)s by "
+msgstr "%(title)s par "
+
+#: bookwyrm/templates/snippets/boost_button.html:8
+#: bookwyrm/templates/snippets/boost_button.html:9
+#: bookwyrm/templates/snippets/status/status_body.html:41
+#: bookwyrm/templates/snippets/status/status_body.html:42
+msgid "Boost status"
+msgstr "Partager le statut"
+
+#: bookwyrm/templates/snippets/boost_button.html:16
+#: bookwyrm/templates/snippets/boost_button.html:17
+msgid "Un-boost status"
+msgstr "Annuler le partage du statut"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:3
+msgid "Spoiler alert:"
+msgstr "Alerte Spoiler :"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:4
+msgid "Spoilers ahead!"
+msgstr "Attention spoilers !"
+
+#: bookwyrm/templates/snippets/create_status.html:9
+msgid "Review"
+msgstr "Critique"
+
+#: bookwyrm/templates/snippets/create_status.html:12
+#: bookwyrm/templates/snippets/create_status_form.html:44
+msgid "Comment"
+msgstr "Commentaire"
+
+#: bookwyrm/templates/snippets/create_status.html:15
+msgid "Quote"
+msgstr "Citation"
+
+#: bookwyrm/templates/snippets/create_status_form.html:21
+#: bookwyrm/templates/snippets/shelf.html:17
+msgid "Rating"
+msgstr "Note"
+
+#: bookwyrm/templates/snippets/create_status_form.html:23
+#: bookwyrm/templates/snippets/rate_action.html:14
+#: bookwyrm/templates/snippets/stars.html:3
+msgid "No rating"
+msgstr "Aucune note"
+
+#: bookwyrm/templates/snippets/create_status_form.html:54
+msgid "Include spoiler alert"
+msgstr "Afficher une alerte spoiler"
+
+#: bookwyrm/templates/snippets/create_status_form.html:60
+#: bookwyrm/templates/snippets/privacy-icons.html:15
+#: bookwyrm/templates/snippets/privacy-icons.html:16
+#: bookwyrm/templates/snippets/privacy_select.html:19
+msgid "Private"
+msgstr "Privé"
+
+#: bookwyrm/templates/snippets/create_status_form.html:67
+msgid "Post"
+msgstr "Publier"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:4
+msgid "Delete these read dates?"
+msgstr "Supprimer ces dates de lecture ?"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:7
+#, python-format
+msgid "You are deleting this readthrough and its %(count)s associated progress updates."
+msgstr "Vous avez supprimé ce résumé et ses %(count)s progressions associées."
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:15
+#: bookwyrm/templates/snippets/follow_request_buttons.html:13
+msgid "Delete"
+msgstr "Supprimer"
+
+#: bookwyrm/templates/snippets/fav_button.html:7
+#: bookwyrm/templates/snippets/fav_button.html:8
+#: bookwyrm/templates/snippets/status/status_body.html:45
+#: bookwyrm/templates/snippets/status/status_body.html:46
+msgid "Like status"
+msgstr "Ajouter le statut aux favoris"
+
+#: bookwyrm/templates/snippets/fav_button.html:15
+#: bookwyrm/templates/snippets/fav_button.html:16
+msgid "Un-like status"
+msgstr "Supprimer le statut des favoris"
+
+#: bookwyrm/templates/snippets/follow_button.html:6
+msgid "Follow request already sent."
+msgstr "Demande d’abonnement déjà envoyée"
+
+#: bookwyrm/templates/snippets/follow_button.html:19
+msgid "Send follow request"
+msgstr "Envoyer une demande d’abonnement"
+
+#: bookwyrm/templates/snippets/follow_button.html:21
+msgid "Follow"
+msgstr "S’abonner"
+
+#: bookwyrm/templates/snippets/follow_button.html:27
+msgid "Unfollow"
+msgstr "Se désabonner"
+
+#: bookwyrm/templates/snippets/follow_request_buttons.html:8
+msgid "Accept"
+msgstr "Accepter"
+
+#: bookwyrm/templates/snippets/generated_status/goal.html:1
+#, python-format
+msgid "set a goal to read %(counter)s book in %(year)s"
+msgid_plural "set a goal to read %(counter)s books in %(year)s"
+msgstr[0] "souhaite lire %(counter)s livre en %(year)s"
+msgstr[1] "souhaite lire %(counter)s livres en %(year)s"
+
+#: bookwyrm/templates/snippets/goal_card.html:21
+msgid "Dismiss message"
+msgstr "Rejeter le message"
+
+#: bookwyrm/templates/snippets/goal_card.html:22
+#, python-format
+msgid "You can set or change your reading goal any time from your profile page "
+msgstr "Vous pouvez définir ou changer vore défi lecture à n’importe quel moment depuis votre profil "
+
+#: bookwyrm/templates/snippets/goal_form.html:9
+msgid "Reading goal:"
+msgstr "Défi lecture :"
+
+#: bookwyrm/templates/snippets/goal_form.html:14
+msgid "books"
+msgstr "livres"
+
+#: bookwyrm/templates/snippets/goal_form.html:19
+msgid "Goal privacy:"
+msgstr "Confidentialité du défi :"
+
+#: bookwyrm/templates/snippets/goal_form.html:26
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:37
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:29
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:20
+msgid "Post to feed"
+msgstr "Publier sur le fil d’actualité"
+
+#: bookwyrm/templates/snippets/goal_form.html:30
+msgid "Set goal"
+msgstr "Valider ce défi"
+
+#: bookwyrm/templates/snippets/goal_progress.html:5
+msgid "Success!"
+msgstr "Bravo !"
+
+#: bookwyrm/templates/snippets/goal_progress.html:7
+#, python-format
+msgid "%(percent)s%% complete!"
+msgstr "%(percent)s%% terminé !"
+
+#: bookwyrm/templates/snippets/goal_progress.html:10
+#, python-format
+msgid "You've read %(read_count)s of %(goal_count)s books ."
+msgstr "Vous avez lu %(read_count)s sur %(goal_count)s livres ."
+
+#: bookwyrm/templates/snippets/goal_progress.html:12
+#, python-format
+msgid "%(username)s has read %(read_count)s of %(goal_count)s books ."
+msgstr "%(username)s a lu %(read_count)s sur %(goal_count)s livres ."
+
+#: bookwyrm/templates/snippets/pagination.html:7
+msgid "Previous"
+msgstr "Précédente"
+
+#: bookwyrm/templates/snippets/pagination.html:15
+msgid "Next"
+msgstr "Suivante"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:3
+#: bookwyrm/templates/snippets/privacy-icons.html:4
+#: bookwyrm/templates/snippets/privacy_select.html:10
+msgid "Public"
+msgstr "Public"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:7
+#: bookwyrm/templates/snippets/privacy-icons.html:8
+#: bookwyrm/templates/snippets/privacy_select.html:13
+msgid "Unlisted"
+msgstr "Non listé"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:12
+msgid "Followers-only"
+msgstr "Abonnemé(e)s uniquement"
+
+#: bookwyrm/templates/snippets/privacy_select.html:6
+msgid "Post privacy"
+msgstr "Confidentialité du statut"
+
+#: bookwyrm/templates/snippets/privacy_select.html:16
+#: bookwyrm/templates/user/followers.html:13
+msgid "Followers"
+msgstr "Abonnements"
+
+#: bookwyrm/templates/snippets/progress_update.html:6
+msgid "Progress:"
+msgstr "Progression :"
+
+#: bookwyrm/templates/snippets/progress_update.html:16
+#: bookwyrm/templates/snippets/readthrough_form.html:22
+msgid "pages"
+msgstr "pages"
+
+#: bookwyrm/templates/snippets/progress_update.html:17
+#: bookwyrm/templates/snippets/readthrough_form.html:23
+msgid "percent"
+msgstr "pourcent"
+
+#: bookwyrm/templates/snippets/progress_update.html:25
+#, python-format
+msgid "of %(book.pages)s pages"
+msgstr "sur %(book.pages)s pages"
+
+#: bookwyrm/templates/snippets/rate_action.html:4
+msgid "Leave a rating"
+msgstr "Laisser une note"
+
+#: bookwyrm/templates/snippets/rate_action.html:29
+msgid "Rate"
+msgstr "Noter"
+
+#: bookwyrm/templates/snippets/readthrough.html:7
+msgid "Progress Updates:"
+msgstr "Progression :"
+
+#: bookwyrm/templates/snippets/readthrough.html:11
+msgid "finished"
+msgstr "terminé"
+
+#: bookwyrm/templates/snippets/readthrough.html:14
+msgid "Show all updates"
+msgstr "Montrer toutes les progressions"
+
+#: bookwyrm/templates/snippets/readthrough.html:30
+msgid "Delete this progress update"
+msgstr "Supprimer cette mise à jour"
+
+#: bookwyrm/templates/snippets/readthrough.html:40
+msgid "started"
+msgstr "commencé"
+
+#: bookwyrm/templates/snippets/readthrough.html:46
+#: bookwyrm/templates/snippets/readthrough.html:60
+msgid "Edit read dates"
+msgstr "Modifier les date de lecture"
+
+#: bookwyrm/templates/snippets/readthrough.html:50
+#, fuzzy
+#| msgid "Delete these read dates?"
+msgid "Delete these read dates"
+msgstr "Supprimer ces dates de lecture"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:7
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:19
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:17
+msgid "Started reading"
+msgstr "Lecture commencée le"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:14
+msgid "Progress"
+msgstr "Progression"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:30
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:25
+msgid "Finished reading"
+msgstr "Lecture terminée le"
+
+#: bookwyrm/templates/snippets/register_form.html:32
+msgid "Sign Up"
+msgstr "S’enregistrer"
+
+#: bookwyrm/templates/snippets/rss_title.html:5
+#: bookwyrm/templates/snippets/status/status_header.html:11
+msgid "rated"
+msgstr "a noté"
+
+#: bookwyrm/templates/snippets/rss_title.html:7
+#: bookwyrm/templates/snippets/status/status_header.html:13
+msgid "reviewed"
+msgstr "a écrit une critique de"
+
+#: bookwyrm/templates/snippets/rss_title.html:9
+#: bookwyrm/templates/snippets/status/status_header.html:15
+msgid "commented on"
+msgstr "a commenté"
+
+#: bookwyrm/templates/snippets/rss_title.html:11
+#: bookwyrm/templates/snippets/status/status_header.html:17
+msgid "quoted"
+msgstr "a cité"
+
+#: bookwyrm/templates/snippets/search_result_text.html:3
+#, python-format
+msgid "by %(author)s"
+msgstr "par %(author)s"
+
+#: bookwyrm/templates/snippets/shelf.html:12
+msgid "Published"
+msgstr "Publié"
+
+#: bookwyrm/templates/snippets/shelf.html:13
+msgid "Shelved"
+msgstr "Ajouté à une étagère"
+
+#: bookwyrm/templates/snippets/shelf.html:14
+msgid "Started"
+msgstr "Commencé"
+
+#: bookwyrm/templates/snippets/shelf.html:15
+msgid "Finished"
+msgstr "Terminé"
+
+#: bookwyrm/templates/snippets/shelf.html:16
+msgid "External links"
+msgstr "Liens externes"
+
+#: bookwyrm/templates/snippets/shelf.html:44
+msgid "OpenLibrary"
+msgstr "OpenLibrary"
+
+#: bookwyrm/templates/snippets/shelf.html:61
+msgid "This shelf is empty."
+msgstr "Cette étagère est vide"
+
+#: bookwyrm/templates/snippets/shelf.html:67
+msgid "Delete shelf"
+msgstr "Supprimer l’étagère"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:4
+msgid "Change shelf"
+msgstr "Changer d’étagère"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:27
+msgid "Unshelve"
+msgstr "Enlever de l’étagère"
+
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:5
+#, python-format
+msgid "Finish \"%(book_title)s \""
+msgstr "Terminer « %(book_title)s »"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_dropdown.html:5
+msgid "More shelves"
+msgstr "Plus d’étagères"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:8
+#, fuzzy
+#| msgid "Started reading"
+msgid "Start reading"
+msgstr "Commencer la lecture"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:13
+#, fuzzy
+#| msgid "Finished reading"
+msgid "Finish reading"
+msgstr "Terminer la lecture"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:16
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
+msgid "Want to read"
+msgstr "Je veux le lire"
+
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:5
+#, python-format
+msgid "Start \"%(book_title)s \""
+msgstr "Commencer « %(book_title)s »"
+
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:5
+#, python-format
+msgid "Want to Read \"%(book_title)s \""
+msgstr "A envie de lire « %(book_title)s »"
+
+#: bookwyrm/templates/snippets/status/status.html:9
+msgid "boosted"
+msgstr "partagé"
+
+#: bookwyrm/templates/snippets/status/status_body.html:24
+#: bookwyrm/templates/snippets/status/status_body.html:37
+#: bookwyrm/templates/snippets/status/status_body.html:38
+msgid "Reply"
+msgstr "Répondre"
+
+#: bookwyrm/templates/snippets/status/status_content.html:18
+#: bookwyrm/templates/snippets/trimmed_text.html:15
+msgid "Show more"
+msgstr "Déplier"
+
+#: bookwyrm/templates/snippets/status/status_content.html:25
+#: bookwyrm/templates/snippets/trimmed_text.html:25
+msgid "Show less"
+msgstr "Replier"
+
+#: bookwyrm/templates/snippets/status/status_content.html:46
+msgid "Open image in new window"
+msgstr "Ouvrir l’image dans une nouvelle fenêtre"
+
+#: bookwyrm/templates/snippets/status/status_header.html:22
+#, fuzzy, python-format
+#| msgid "favorited your %(preview_name)s "
+msgid "replied to %(username)s's review "
+msgstr "Messages directs avec %(username)s "
+
+#: bookwyrm/templates/snippets/status/status_header.html:24
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's comment "
+msgstr "a répondu à votre statut "
+
+#: bookwyrm/templates/snippets/status/status_header.html:26
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's quote "
+msgstr "a répondu à votre statut "
+
+#: bookwyrm/templates/snippets/status/status_header.html:28
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's status "
+msgstr "a répondu à votre statut "
+
+#: bookwyrm/templates/snippets/status/status_options.html:7
+#: bookwyrm/templates/snippets/user_options.html:7
+msgid "More options"
+msgstr "Plus d’options"
+
+#: bookwyrm/templates/snippets/status/status_options.html:17
+msgid "Delete status"
+msgstr "Supprimer le statut"
+
+#: bookwyrm/templates/snippets/status/status_options.html:23
+#: bookwyrm/templates/snippets/user_options.html:13
+msgid "Send direct message"
+msgstr "Envoyer un message direct"
+
+#: bookwyrm/templates/snippets/switch_edition_button.html:5
+msgid "Switch to this edition"
+msgstr "Changer vers cette édition"
+
+#: bookwyrm/templates/snippets/tag.html:14
+msgid "Remove tag"
+msgstr "Supprimer le tag"
+
+#: bookwyrm/templates/tag.html:9
+#, python-format
+msgid "Books tagged \"%(tag.name)s\""
+msgstr "Livres tagués « %(tag.name)s »"
+
+#: bookwyrm/templates/user/create_shelf_form.html:5
+#: bookwyrm/templates/user/create_shelf_form.html:22
+msgid "Create Shelf"
+msgstr "Créer l’étagère"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:5
+msgid "Edit Shelf"
+msgstr "Modifier l’étagère"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:26
+msgid "Update shelf"
+msgstr "Mettre l’étagère à jour"
+
+#: bookwyrm/templates/user/followers.html:7
+#: bookwyrm/templates/user/following.html:7 bookwyrm/templates/user/user.html:9
+msgid "User Profile"
+msgstr "Profil"
+
+#: bookwyrm/templates/user/followers.html:29
+#, python-format
+msgid "%(username)s has no followers"
+msgstr "%(username)s n’a pas d’abonné(e)"
+
+#: bookwyrm/templates/user/following.html:13
+msgid "Following"
+msgstr "Abonné(e) à"
+
+#: bookwyrm/templates/user/following.html:29
+#, python-format
+msgid "%(username)s isn't following any users"
+msgstr "%(username)s n’est abonné(e) à personne"
+
+#: bookwyrm/templates/user/lists.html:9
+msgid "Your Lists"
+msgstr "Vos listes"
+
+#: bookwyrm/templates/user/lists.html:11
+#, fuzzy, python-format
+#| msgid "Join %(name)s"
+msgid "Lists: %(username)s"
+msgstr "Listes : %(username)s"
+
+#: bookwyrm/templates/user/lists.html:17 bookwyrm/templates/user/lists.html:29
+msgid "Create list"
+msgstr "Créer une liste"
+
+#: bookwyrm/templates/user/shelf.html:9
+msgid "Your Shelves"
+msgstr "Vos étagères"
+
+#: bookwyrm/templates/user/shelf.html:11
+#, python-format
+msgid "%(username)s: Shelves"
+msgstr "%(username)s : Étagères"
+
+#: bookwyrm/templates/user/shelf.html:33
+msgid "Create shelf"
+msgstr "Créer l’étagère"
+
+#: bookwyrm/templates/user/shelf.html:54
+msgid "Edit shelf"
+msgstr "Modifier l’étagère"
+
+#: bookwyrm/templates/user/user.html:15
+msgid "Edit profile"
+msgstr "Modifier le profil"
+
+#: bookwyrm/templates/user/user.html:26
+#: bookwyrm/templates/user/user_layout.html:68
+msgid "Shelves"
+msgstr "Étagères"
+
+#: bookwyrm/templates/user/user.html:31
+#, python-format
+msgid "See all %(size)s"
+msgstr "Voir les %(size)s"
+
+#: bookwyrm/templates/user/user.html:44
+#, python-format
+msgid "See all %(shelf_count)s shelves"
+msgstr "Voir les %(shelf_count)s étagères"
+
+#: bookwyrm/templates/user/user.html:56
+#, python-format
+msgid "Set a reading goal for %(year)s"
+msgstr "Définir un défi lecture pour %(year)s"
+
+#: bookwyrm/templates/user/user.html:62
+msgid "User Activity"
+msgstr "Activité du compte"
+
+#: bookwyrm/templates/user/user.html:65
+msgid "RSS feed"
+msgstr "Flux RSS"
+
+#: bookwyrm/templates/user/user.html:76
+msgid "No activities yet!"
+msgstr "Aucune activité pour l’instant !"
+
+#: bookwyrm/templates/user/user_layout.html:32
+msgid "Follow Requests"
+msgstr "Demandes d’abonnement"
+
+#: bookwyrm/templates/user/user_layout.html:50
+msgid "Activity"
+msgstr "Activité"
+
+#: bookwyrm/templates/user/user_layout.html:56
+msgid "Reading Goal"
+msgstr "Défi lecture"
+
+#: bookwyrm/templates/user/user_preview.html:13
+#, python-format
+msgid "Joined %(date)s"
+msgstr "Enregistré(e) %(date)s"
+
+#: bookwyrm/templates/user/user_preview.html:15
+#, fuzzy, python-format
+#| msgid "%(username)s has no followers"
+msgid "%(counter)s follower"
+msgid_plural "%(counter)s followers"
+msgstr[0] "%(username)s n’a pas d’abonné(e)"
+msgstr[1] "%(username)s n’a pas d’abonné(e)s"
+
+#: bookwyrm/templates/user/user_preview.html:16
+#, python-format
+msgid "%(counter)s following"
+msgstr "%(counter)s abonnements"
+
+#~ msgid "Created and curated by"
+#~ msgstr "Créée et modérée par"
+
+#~ msgid "Created by"
+#~ msgstr "Créée par"
+
+#~ msgid "Create New Shelf"
+#~ msgstr "Créer une nouvelle étagère"
+
+#, fuzzy
+#~| msgid "Create list"
+#~ msgid "Create new list"
+#~ msgstr "Créer une nouvelle liste"
+
+#~ msgid "Added by"
+#~ msgstr "Ajouté par"
+
+#~ msgid "added"
+#~ msgstr "a ajouté"
+
+#~ msgid "suggested adding"
+#~ msgstr "a suggéré l’ajout de"
+
+#~ msgid "Change password"
+#~ msgstr "Changer le mot de passe"
+
+#~ msgid "Blocked users"
+#~ msgstr "Comptes bloqués"
+
+#~ msgid "Comment:"
+#~ msgstr "Commentaire :"
+
+#~ msgid "%(year)s reading goal"
+#~ msgstr "Défi lecture pour %(year)s"
diff --git a/locale/zh_CN/LC_MESSAGES/django.mo b/locale/zh_CN/LC_MESSAGES/django.mo
new file mode 100644
index 00000000..96663265
Binary files /dev/null and b/locale/zh_CN/LC_MESSAGES/django.mo differ
diff --git a/locale/zh_CN/LC_MESSAGES/django.po b/locale/zh_CN/LC_MESSAGES/django.po
new file mode 100644
index 00000000..34dbed2a
--- /dev/null
+++ b/locale/zh_CN/LC_MESSAGES/django.po
@@ -0,0 +1,1881 @@
+# Simplified Chinese language text for the BookWyrm UI
+# Copyright (C) 2021 Mouse Reeve
+# This file is distributed under the same license as the bookwyrm package.
+# Mouse Reeve , 2021.
+#
+#, fuzzy
+msgid ""
+msgstr ""
+"Project-Id-Version: 0.1.1\n"
+"Report-Msgid-Bugs-To: \n"
+"POT-Creation-Date: 2021-03-07 23:40+0000\n"
+"PO-Revision-Date: 2021-03-02 10:35+0000\n"
+"Last-Translator: Kana \n"
+"Language-Team: Mouse Reeve \n"
+"Language: zh_CN\n"
+"MIME-Version: 1.0\n"
+"Content-Type: text/plain; charset=UTF-8\n"
+"Content-Transfer-Encoding: 8bit\n"
+"Plural-Forms: nplurals=1; plural=0;\n"
+
+#: bookwyrm/forms.py:185
+msgid "One Day"
+msgstr ""
+
+#: bookwyrm/forms.py:186
+msgid "One Week"
+msgstr ""
+
+#: bookwyrm/forms.py:187
+msgid "One Month"
+msgstr ""
+
+#: bookwyrm/forms.py:188
+msgid "Does Not Expire"
+msgstr ""
+
+#: bookwyrm/forms.py:190
+#, python-format
+msgid "%(count)d uses"
+msgstr ""
+
+#: bookwyrm/forms.py:192
+#, fuzzy
+#| msgid "Unlisted"
+msgid "Unlimited"
+msgstr "不公开"
+
+#: bookwyrm/models/fields.py:24
+#, python-format
+msgid "%(value)s is not a valid remote_id"
+msgstr ""
+
+#: bookwyrm/models/fields.py:33 bookwyrm/models/fields.py:42
+#, python-format
+msgid "%(value)s is not a valid username"
+msgstr ""
+
+#: bookwyrm/models/fields.py:164 bookwyrm/templates/layout.html:152
+msgid "username"
+msgstr "用户名"
+
+#: bookwyrm/models/fields.py:169
+msgid "A user with that username already exists."
+msgstr ""
+
+#: bookwyrm/settings.py:142
+msgid "English"
+msgstr ""
+
+#: bookwyrm/settings.py:143
+msgid "German"
+msgstr ""
+
+#: bookwyrm/settings.py:144
+msgid "Spanish"
+msgstr ""
+
+#: bookwyrm/settings.py:145
+msgid "French"
+msgstr ""
+
+#: bookwyrm/settings.py:146
+msgid "Simplified Chinese"
+msgstr ""
+
+#: bookwyrm/templates/author.html:16 bookwyrm/templates/author.html:17
+#: bookwyrm/templates/edit_author.html:5
+msgid "Edit Author"
+msgstr "编辑作者"
+
+#: bookwyrm/templates/author.html:32
+msgid "Wikipedia"
+msgstr "维基百科"
+
+#: bookwyrm/templates/author.html:37
+#, python-format
+msgid "Books by %(name)s"
+msgstr "%(name)s 所著的书"
+
+#: bookwyrm/templates/book.html:21
+#: bookwyrm/templates/discover/large-book.html:12
+#: bookwyrm/templates/discover/small-book.html:9
+msgid "by"
+msgstr ""
+
+#: bookwyrm/templates/book.html:29 bookwyrm/templates/book.html:30
+#: bookwyrm/templates/edit_book.html:5
+msgid "Edit Book"
+msgstr "编辑书目"
+
+#: bookwyrm/templates/book.html:45
+msgid "Add cover"
+msgstr "添加封面"
+
+#: bookwyrm/templates/book.html:51 bookwyrm/templates/lists/list.html:89
+msgid "Add"
+msgstr "添加"
+
+#: bookwyrm/templates/book.html:60
+msgid "ISBN:"
+msgstr "ISBN:"
+
+#: bookwyrm/templates/book.html:67 bookwyrm/templates/edit_book.html:107
+msgid "OCLC Number:"
+msgstr "OCLC 号:"
+
+#: bookwyrm/templates/book.html:74 bookwyrm/templates/edit_book.html:111
+msgid "ASIN:"
+msgstr "ASIN:"
+
+#: bookwyrm/templates/book.html:84
+#, fuzzy, python-format
+#| msgid "of %(book.pages)s pages"
+msgid "%(format)s, %(pages)s pages"
+msgstr "全书 %(book.pages)s 页"
+
+#: bookwyrm/templates/book.html:86
+#, fuzzy, python-format
+#| msgid "of %(book.pages)s pages"
+msgid "%(pages)s pages"
+msgstr "全书 %(book.pages)s 页"
+
+#: bookwyrm/templates/book.html:91
+msgid "View on OpenLibrary"
+msgstr "在 OpenLibrary 查看"
+
+#: bookwyrm/templates/book.html:100
+#, python-format
+msgid "(%(review_count)s review)"
+msgid_plural "(%(review_count)s reviews)"
+msgstr[0] ""
+
+#: bookwyrm/templates/book.html:106
+msgid "Add Description"
+msgstr "添加描述"
+
+#: bookwyrm/templates/book.html:113 bookwyrm/templates/edit_book.html:39
+#: bookwyrm/templates/lists/form.html:12
+msgid "Description:"
+msgstr "描述:"
+
+#: bookwyrm/templates/book.html:117 bookwyrm/templates/edit_author.html:78
+#: bookwyrm/templates/edit_book.html:120 bookwyrm/templates/lists/form.html:42
+#: bookwyrm/templates/preferences/edit_user.html:50
+#: bookwyrm/templates/settings/site.html:89
+#: bookwyrm/templates/snippets/progress_update.html:21
+#: bookwyrm/templates/snippets/readthrough.html:64
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:42
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:34
+msgid "Save"
+msgstr "保存"
+
+#: bookwyrm/templates/book.html:118 bookwyrm/templates/book.html:167
+#: bookwyrm/templates/edit_author.html:79 bookwyrm/templates/edit_book.html:121
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:17
+#: bookwyrm/templates/snippets/goal_form.html:32
+#: bookwyrm/templates/snippets/readthrough.html:65
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:43
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:35
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:28
+msgid "Cancel"
+msgstr "取消"
+
+#: bookwyrm/templates/book.html:127
+#, fuzzy, python-format
+#| msgid "%(title)s by "
+msgid "%(count)s editions "
+msgstr "%(title)s 来自"
+
+#: bookwyrm/templates/book.html:135
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "This edition is on your %(shelf_name)s shelf."
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/book.html:141
+#, fuzzy, python-format
+#| msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgid "A different edition of this book is on your %(shelf_name)s shelf."
+msgstr " 添加了 %(book_title)s 到你的列表 \"%(list_name)s \""
+
+#: bookwyrm/templates/book.html:150
+msgid "Your reading activity"
+msgstr "你的阅读活动"
+
+#: bookwyrm/templates/book.html:152
+msgid "Add read dates"
+msgstr "添加阅读日期"
+
+#: bookwyrm/templates/book.html:157
+msgid "You don't have any reading activity for this book."
+msgstr "你还没有任何这本书的阅读活动。"
+
+#: bookwyrm/templates/book.html:164
+msgid "Create"
+msgstr "创建"
+
+#: bookwyrm/templates/book.html:186
+msgid "Tags"
+msgstr "标签"
+
+#: bookwyrm/templates/book.html:190 bookwyrm/templates/snippets/tag.html:18
+msgid "Add tag"
+msgstr "添加标签"
+
+#: bookwyrm/templates/book.html:207
+msgid "Subjects"
+msgstr "主题"
+
+#: bookwyrm/templates/book.html:218
+msgid "Places"
+msgstr "地点"
+
+#: bookwyrm/templates/book.html:229 bookwyrm/templates/layout.html:64
+#: bookwyrm/templates/lists/lists.html:4 bookwyrm/templates/lists/lists.html:9
+#: bookwyrm/templates/search_results.html:92
+#: bookwyrm/templates/user/user_layout.html:62
+msgid "Lists"
+msgstr "列表"
+
+#: bookwyrm/templates/book.html:258
+msgid "rated it"
+msgstr "评价了"
+
+#: bookwyrm/templates/components/inline_form.html:8
+#: bookwyrm/templates/feed/feed_layout.html:54
+msgid "Close"
+msgstr "关闭"
+
+#: bookwyrm/templates/discover/about.html:7
+#, python-format
+msgid "About %(site_name)s"
+msgstr "关于 %(site_name)s"
+
+#: bookwyrm/templates/discover/about.html:10
+#: bookwyrm/templates/discover/about.html:20
+msgid "Code of Conduct"
+msgstr "行为准则"
+
+#: bookwyrm/templates/discover/about.html:13
+#: bookwyrm/templates/discover/about.html:29
+msgid "Privacy Policy"
+msgstr "隐私政策"
+
+#: bookwyrm/templates/discover/discover.html:6
+msgid "Recent Books"
+msgstr "最近书目"
+
+#: bookwyrm/templates/discover/landing_layout.html:5
+msgid "Welcome"
+msgstr "欢迎"
+
+#: bookwyrm/templates/discover/landing_layout.html:17
+msgid "Decentralized"
+msgstr "去中心化"
+
+#: bookwyrm/templates/discover/landing_layout.html:23
+msgid "Friendly"
+msgstr "友好"
+
+#: bookwyrm/templates/discover/landing_layout.html:29
+msgid "Anti-Corporate"
+msgstr "反企业"
+
+#: bookwyrm/templates/discover/landing_layout.html:44
+#, python-format
+msgid "Join %(name)s"
+msgstr "加入 %(name)s"
+
+#: bookwyrm/templates/discover/landing_layout.html:49
+#: bookwyrm/templates/login.html:48
+msgid "This instance is closed"
+msgstr "本实例不开放。"
+
+#: bookwyrm/templates/discover/landing_layout.html:55
+msgid "Your Account"
+msgstr "你的帐号"
+
+#: bookwyrm/templates/edit_author.html:13 bookwyrm/templates/edit_book.html:13
+msgid "Added:"
+msgstr "添加了:"
+
+#: bookwyrm/templates/edit_author.html:14 bookwyrm/templates/edit_book.html:14
+msgid "Updated:"
+msgstr "更新了:"
+
+#: bookwyrm/templates/edit_author.html:15 bookwyrm/templates/edit_book.html:15
+msgid "Last edited by:"
+msgstr "最后编辑人:"
+
+#: bookwyrm/templates/edit_author.html:31 bookwyrm/templates/edit_book.html:30
+msgid "Metadata"
+msgstr "元数据"
+
+#: bookwyrm/templates/edit_author.html:32 bookwyrm/templates/lists/form.html:8
+#: bookwyrm/templates/user/create_shelf_form.html:13
+#: bookwyrm/templates/user/edit_shelf_form.html:14
+msgid "Name:"
+msgstr "名称:"
+
+#: bookwyrm/templates/edit_author.html:37
+msgid "Bio:"
+msgstr "简介:"
+
+#: bookwyrm/templates/edit_author.html:42
+msgid "Wikipedia link:"
+msgstr "维基百科链接:"
+
+#: bookwyrm/templates/edit_author.html:47
+msgid "Birth date:"
+msgstr "出生日期:"
+
+#: bookwyrm/templates/edit_author.html:52
+msgid "Death date:"
+msgstr "死亡日期:"
+
+#: bookwyrm/templates/edit_author.html:58
+msgid "Author Identifiers"
+msgstr "作者标识号:"
+
+#: bookwyrm/templates/edit_author.html:59 bookwyrm/templates/edit_book.html:103
+msgid "Openlibrary key:"
+msgstr "Openlibrary key:"
+
+#: bookwyrm/templates/edit_author.html:64
+msgid "Librarything key:"
+msgstr "Librarything key:"
+
+#: bookwyrm/templates/edit_author.html:69
+msgid "Goodreads key:"
+msgstr "Goodreads key:"
+
+#: bookwyrm/templates/edit_book.html:31
+msgid "Title:"
+msgstr "标题:"
+
+#: bookwyrm/templates/edit_book.html:35
+msgid "Subtitle:"
+msgstr "副标题:"
+
+#: bookwyrm/templates/edit_book.html:43
+msgid "Series:"
+msgstr "系列:"
+
+#: bookwyrm/templates/edit_book.html:47
+msgid "Series number:"
+msgstr "系列编号:"
+
+#: bookwyrm/templates/edit_book.html:51
+msgid "First published date:"
+msgstr "初版时间:"
+
+#: bookwyrm/templates/edit_book.html:55
+msgid "Published date:"
+msgstr "出版时间:"
+
+#: bookwyrm/templates/edit_book.html:68
+#: bookwyrm/templates/snippets/shelf.html:9
+msgid "Cover"
+msgstr "封面"
+
+#: bookwyrm/templates/edit_book.html:78
+msgid "Physical Properties"
+msgstr "实体性质"
+
+#: bookwyrm/templates/edit_book.html:79
+msgid "Format:"
+msgstr "格式:"
+
+#: bookwyrm/templates/edit_book.html:87
+msgid "Pages:"
+msgstr "页数:"
+
+#: bookwyrm/templates/edit_book.html:94
+msgid "Book Identifiers"
+msgstr "书目标识号"
+
+#: bookwyrm/templates/edit_book.html:95
+msgid "ISBN 13:"
+msgstr "ISBN 13:"
+
+#: bookwyrm/templates/edit_book.html:99
+msgid "ISBN 10:"
+msgstr "ISBN 10:"
+
+#: bookwyrm/templates/editions.html:5
+#, python-format
+msgid "Editions of %(book_title)s"
+msgstr "%(book_title)s 的各版本"
+
+#: bookwyrm/templates/editions.html:9
+#, python-format
+msgid "Editions of \"%(work_title)s\" "
+msgstr "\"%(work_title)s\" 的各版本"
+
+#: bookwyrm/templates/error.html:4
+msgid "Oops!"
+msgstr "哎呀!"
+
+#: bookwyrm/templates/error.html:8
+msgid "Server Error"
+msgstr "服务器错误"
+
+#: bookwyrm/templates/error.html:9
+msgid "Something went wrong! Sorry about that."
+msgstr "某些东西出错了!对不起啦。"
+
+#: bookwyrm/templates/feed/direct_messages.html:8
+#, python-format
+msgid "Direct Messages with %(username)s "
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/feed/direct_messages.html:10
+#: bookwyrm/templates/layout.html:87
+msgid "Direct Messages"
+msgstr "私信"
+
+#: bookwyrm/templates/feed/direct_messages.html:13
+msgid "All messages"
+msgstr "所有消息"
+
+#: bookwyrm/templates/feed/direct_messages.html:22
+msgid "You have no messages right now."
+msgstr "你现在没有消息。"
+
+#: bookwyrm/templates/feed/feed.html:6
+#, python-format
+msgid "%(tab_title)s Timeline"
+msgstr "%(tab_title)s 时间线"
+
+#: bookwyrm/templates/feed/feed.html:10 bookwyrm/views/feed.py:33
+msgid "Home"
+msgstr "主页"
+
+#: bookwyrm/templates/feed/feed.html:13 bookwyrm/views/feed.py:37
+msgid "Local"
+msgstr "本站"
+
+#: bookwyrm/templates/feed/feed.html:16 bookwyrm/views/feed.py:41
+msgid "Federated"
+msgstr "跨站"
+
+#: bookwyrm/templates/feed/feed.html:24
+msgid "Announcements"
+msgstr "公告"
+
+#: bookwyrm/templates/feed/feed.html:32
+msgid "There aren't any activities right now! Try following a user to get started"
+msgstr "现在还没有任何活动!尝试着从关注一个用户开始吧"
+
+#: bookwyrm/templates/feed/feed_layout.html:5
+msgid "Updates"
+msgstr "更新"
+
+#: bookwyrm/templates/feed/feed_layout.html:11
+msgid "Your books"
+msgstr "你的书目"
+
+#: bookwyrm/templates/feed/feed_layout.html:13
+msgid "There are no books here right now! Try searching for a book to get started"
+msgstr "现在这里还没有任何书目!尝试着从搜索某本书开始吧"
+
+#: bookwyrm/templates/feed/feed_layout.html:23
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Read"
+msgid "To Read"
+msgstr "阅读"
+
+#: bookwyrm/templates/feed/feed_layout.html:24
+#: bookwyrm/templates/user/shelf.html:24
+#, fuzzy
+#| msgid "Start reading"
+msgid "Currently Reading"
+msgstr "开始阅读"
+
+#: bookwyrm/templates/feed/feed_layout.html:25
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:11
+#: bookwyrm/templates/user/shelf.html:24
+msgid "Read"
+msgstr "阅读"
+
+#: bookwyrm/templates/feed/feed_layout.html:76 bookwyrm/templates/goal.html:26
+#: bookwyrm/templates/snippets/goal_card.html:6
+#, python-format
+msgid "%(year)s Reading Goal"
+msgstr "%(year)s 阅读目标"
+
+#: bookwyrm/templates/feed/status.html:8
+msgid "Back"
+msgstr "返回"
+
+#: bookwyrm/templates/goal.html:7
+#, python-format
+msgid "%(year)s Reading Progress"
+msgstr "%(year)s 阅读进度"
+
+#: bookwyrm/templates/goal.html:11
+msgid "Edit Goal"
+msgstr "编辑目标"
+
+#: bookwyrm/templates/goal.html:30
+#: bookwyrm/templates/snippets/goal_card.html:13
+#, python-format
+msgid "Set a goal for how many books you'll finish reading in %(year)s, and track your progress throughout the year."
+msgstr "设定一个 %(year)s 内要读多少书的目标,并记录你全年的进度。"
+
+#: bookwyrm/templates/goal.html:39
+#, python-format
+msgid "%(name)s hasn't set a reading goal for %(year)s."
+msgstr "%(name)s 还没有设定 %(year)s 的阅读目标。"
+
+#: bookwyrm/templates/goal.html:51
+#, python-format
+msgid "Your %(year)s Books"
+msgstr "你 %(year)s 的书目"
+
+#: bookwyrm/templates/goal.html:53
+#, python-format
+msgid "%(username)s's %(year)s Books"
+msgstr "%(username)s 在 %(year)s 的书目"
+
+#: bookwyrm/templates/import.html:5 bookwyrm/templates/import.html:9
+#: bookwyrm/templates/layout.html:102
+msgid "Import Books"
+msgstr "导入书目"
+
+#: bookwyrm/templates/import.html:14
+msgid "Data source"
+msgstr "数据来源"
+
+#: bookwyrm/templates/import.html:32
+msgid "Include reviews"
+msgstr "纳入书评"
+
+#: bookwyrm/templates/import.html:37
+msgid "Privacy setting for imported reviews:"
+msgstr "导入书评的隐私设定"
+
+#: bookwyrm/templates/import.html:41
+msgid "Import"
+msgstr "导入"
+
+#: bookwyrm/templates/import.html:46
+msgid "Recent Imports"
+msgstr "最近的导入"
+
+#: bookwyrm/templates/import.html:48
+msgid "No recent imports"
+msgstr "无最近的导入"
+
+#: bookwyrm/templates/import_status.html:6
+#: bookwyrm/templates/import_status.html:10
+msgid "Import Status"
+msgstr "导入状态"
+
+#: bookwyrm/templates/import_status.html:13
+msgid "Import started:"
+msgstr "导入开始:"
+
+#: bookwyrm/templates/import_status.html:17
+msgid "Import completed:"
+msgstr "导入完成:"
+
+#: bookwyrm/templates/import_status.html:20
+msgid "TASK FAILED"
+msgstr "任务失败"
+
+#: bookwyrm/templates/import_status.html:26
+msgid "Import still in progress."
+msgstr "还在导入中。"
+
+#: bookwyrm/templates/import_status.html:28
+msgid "(Hit reload to update!)"
+msgstr "(按下重新加载来更新!)"
+
+#: bookwyrm/templates/import_status.html:35
+msgid "Failed to load"
+msgstr "加载失败"
+
+#: bookwyrm/templates/import_status.html:44
+#, python-format
+msgid "Jump to the bottom of the list to select the %(failed_count)s items which failed to import."
+msgstr ""
+
+#: bookwyrm/templates/import_status.html:79
+msgid "Select all"
+msgstr "全选"
+
+#: bookwyrm/templates/import_status.html:82
+msgid "Retry items"
+msgstr "重试项目"
+
+#: bookwyrm/templates/import_status.html:108
+msgid "Successfully imported"
+msgstr "成功导入了"
+
+#: bookwyrm/templates/import_status.html:112
+#: bookwyrm/templates/lists/curate.html:14
+msgid "Book"
+msgstr "书目"
+
+#: bookwyrm/templates/import_status.html:115
+#: bookwyrm/templates/snippets/create_status_form.html:10
+#: bookwyrm/templates/snippets/shelf.html:10
+msgid "Title"
+msgstr "标题"
+
+#: bookwyrm/templates/import_status.html:118
+#: bookwyrm/templates/snippets/shelf.html:11
+msgid "Author"
+msgstr "作者"
+
+#: bookwyrm/templates/import_status.html:141
+msgid "Imported"
+msgstr "已导入"
+
+#: bookwyrm/templates/invite.html:4 bookwyrm/templates/invite.html:12
+#: bookwyrm/templates/login.html:43
+msgid "Create an Account"
+msgstr "创建帐号"
+
+#: bookwyrm/templates/invite.html:21
+msgid "Permission Denied"
+msgstr "没有权限"
+
+#: bookwyrm/templates/invite.html:22
+msgid "Sorry! This invite code is no longer valid."
+msgstr "抱歉!此邀请码已不再有效。"
+
+#: bookwyrm/templates/isbn_search_results.html:4
+#: bookwyrm/templates/search_results.html:4
+msgid "Search Results"
+msgstr "搜索结果"
+
+#: bookwyrm/templates/isbn_search_results.html:9
+#: bookwyrm/templates/search_results.html:9
+#, python-format
+msgid "Search Results for \"%(query)s\""
+msgstr "\"%(query)s\" 的搜索结果"
+
+#: bookwyrm/templates/isbn_search_results.html:14
+#: bookwyrm/templates/search_results.html:14
+msgid "Matching Books"
+msgstr "匹配的书目"
+
+#: bookwyrm/templates/isbn_search_results.html:17
+#: bookwyrm/templates/search_results.html:17
+#, python-format
+msgid "No books found for \"%(query)s\""
+msgstr "没有找到 \"%(query)s\" 的书目"
+
+#: bookwyrm/templates/layout.html:33
+msgid "Search for a book or user"
+msgstr "搜索书目或用户"
+
+#: bookwyrm/templates/layout.html:37 bookwyrm/templates/layout.html:38
+#: bookwyrm/templates/lists/list.html:62
+msgid "Search"
+msgstr "搜索"
+
+#: bookwyrm/templates/layout.html:47 bookwyrm/templates/layout.html:48
+msgid "Main navigation menu"
+msgstr "主导航菜单"
+
+#: bookwyrm/templates/layout.html:58
+msgid "Your shelves"
+msgstr "你的书架"
+
+#: bookwyrm/templates/layout.html:61
+msgid "Feed"
+msgstr "动态"
+
+#: bookwyrm/templates/layout.html:92
+#: bookwyrm/templates/preferences/preferences_layout.html:14
+msgid "Profile"
+msgstr "个人资料"
+
+#: bookwyrm/templates/layout.html:97
+msgid "Settings"
+msgstr "设置"
+
+#: bookwyrm/templates/layout.html:111
+#: bookwyrm/templates/settings/admin_layout.html:19
+#: bookwyrm/templates/settings/manage_invites.html:3
+msgid "Invites"
+msgstr "邀请"
+
+#: bookwyrm/templates/layout.html:118
+msgid "Site Configuration"
+msgstr "站点配置"
+
+#: bookwyrm/templates/layout.html:125
+msgid "Log out"
+msgstr "登出"
+
+#: bookwyrm/templates/layout.html:133 bookwyrm/templates/layout.html:134
+#: bookwyrm/templates/notifications.html:6
+#: bookwyrm/templates/notifications.html:10
+msgid "Notifications"
+msgstr "通知"
+
+#: bookwyrm/templates/layout.html:151 bookwyrm/templates/layout.html:155
+#: bookwyrm/templates/login.html:17
+#: bookwyrm/templates/snippets/register_form.html:4
+msgid "Username:"
+msgstr "用户名:"
+
+#: bookwyrm/templates/layout.html:156
+msgid "password"
+msgstr "密码"
+
+#: bookwyrm/templates/layout.html:157 bookwyrm/templates/login.html:36
+msgid "Forgot your password?"
+msgstr "忘记了密码?"
+
+#: bookwyrm/templates/layout.html:160 bookwyrm/templates/login.html:10
+#: bookwyrm/templates/login.html:33
+msgid "Log in"
+msgstr "登录"
+
+#: bookwyrm/templates/layout.html:168
+msgid "Join"
+msgstr ""
+
+#: bookwyrm/templates/layout.html:191
+msgid "About this server"
+msgstr "关于本服务器"
+
+#: bookwyrm/templates/layout.html:195
+msgid "Contact site admin"
+msgstr "联系站点管理员"
+
+#: bookwyrm/templates/layout.html:202
+#, python-format
+msgid "Support %(site_name)s on %(support_title)s "
+msgstr ""
+
+#: bookwyrm/templates/layout.html:206
+msgid "BookWyrm is open source software. You can contribute or report issues on GitHub ."
+msgstr "BookWyrm 是开源软件。你可以在GitHub 贡献或报告问题。"
+
+#: bookwyrm/templates/lists/create_form.html:5
+#: bookwyrm/templates/lists/lists.html:17
+msgid "Create List"
+msgstr "创建列表"
+
+#: bookwyrm/templates/lists/created_text.html:5
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Created and curated by %(username)s "
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/lists/created_text.html:7
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Created by %(username)s "
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/lists/curate.html:6
+msgid "Pending Books"
+msgstr "等候中的书目"
+
+#: bookwyrm/templates/lists/curate.html:7
+msgid "Go to list"
+msgstr "前往列表"
+
+#: bookwyrm/templates/lists/curate.html:9
+msgid "You're all set!"
+msgstr "都弄好了!"
+
+#: bookwyrm/templates/lists/curate.html:15
+msgid "Suggested by"
+msgstr "推荐来自"
+
+#: bookwyrm/templates/lists/curate.html:35
+msgid "Approve"
+msgstr "批准"
+
+#: bookwyrm/templates/lists/curate.html:41
+msgid "Discard"
+msgstr "削除"
+
+#: bookwyrm/templates/lists/edit_form.html:5
+#: bookwyrm/templates/lists/list_layout.html:18
+msgid "Edit List"
+msgstr "编辑列表"
+
+#: bookwyrm/templates/lists/form.html:18
+msgid "List curation:"
+msgstr "列表策展:"
+
+#: bookwyrm/templates/lists/form.html:21
+msgid "Closed"
+msgstr "已关闭"
+
+#: bookwyrm/templates/lists/form.html:22
+msgid "Only you can add and remove books to this list"
+msgstr "只有你可以在此列表中添加或移除书目"
+
+#: bookwyrm/templates/lists/form.html:26
+msgid "Curated"
+msgstr "策展"
+
+#: bookwyrm/templates/lists/form.html:27
+msgid "Anyone can suggest books, subject to your approval"
+msgstr "任何人都可以推荐书目、主题让你批准"
+
+#: bookwyrm/templates/lists/form.html:31
+msgid "Open"
+msgstr "开放"
+
+#: bookwyrm/templates/lists/form.html:32
+msgid "Anyone can add books to this list"
+msgstr "任何人都可以向此列表中添加书目"
+
+#: bookwyrm/templates/lists/list.html:17
+msgid "This list is currently empty"
+msgstr "此列表当前是空的"
+
+#: bookwyrm/templates/lists/list.html:35
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "Added by %(username)s "
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/lists/list.html:41
+msgid "Remove"
+msgstr "移除"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Add Books"
+msgstr "添加书目"
+
+#: bookwyrm/templates/lists/list.html:54
+msgid "Suggest Books"
+msgstr "推荐书目"
+
+#: bookwyrm/templates/lists/list.html:58
+msgid "Search for a book"
+msgstr "搜索书目"
+
+#: bookwyrm/templates/lists/list.html:63
+msgid "search"
+msgstr "搜索"
+
+#: bookwyrm/templates/lists/list.html:69
+msgid "Clear search"
+msgstr "清除搜索"
+
+#: bookwyrm/templates/lists/list.html:74
+#, python-format
+msgid "No books found matching the query \"%(query)s\""
+msgstr "没有符合 \"%(query)s\" 请求的书目"
+
+#: bookwyrm/templates/lists/list.html:75
+msgid "No books found"
+msgstr "没有找到书目"
+
+#: bookwyrm/templates/lists/list.html:89
+msgid "Suggest"
+msgstr "推荐"
+
+#: bookwyrm/templates/lists/lists.html:14
+msgid "Your lists"
+msgstr "你的列表"
+
+#: bookwyrm/templates/lists/lists.html:32
+#, fuzzy, python-format
+#| msgid "See all %(size)s"
+msgid "See all %(size)s lists"
+msgstr "查看所有 %(size)s"
+
+#: bookwyrm/templates/lists/lists.html:40
+msgid "Recent Lists"
+msgstr "最近的列表"
+
+#: bookwyrm/templates/login.html:4
+msgid "Login"
+msgstr "登录"
+
+#: bookwyrm/templates/login.html:23 bookwyrm/templates/password_reset.html:17
+#: bookwyrm/templates/snippets/register_form.html:22
+msgid "Password:"
+msgstr "密码:"
+
+#: bookwyrm/templates/login.html:49
+msgid "Contact an administrator to get an invite"
+msgstr "联系管理员以取得邀请"
+
+#: bookwyrm/templates/login.html:59
+msgid "More about this site"
+msgstr "关于本站点的更多"
+
+#: bookwyrm/templates/notfound.html:4 bookwyrm/templates/notfound.html:8
+msgid "Not Found"
+msgstr "未找到"
+
+#: bookwyrm/templates/notfound.html:9
+msgid "The page you requested doesn't seem to exist!"
+msgstr "你请求的页面似乎并不存在!"
+
+#: bookwyrm/templates/notifications.html:14
+msgid "Delete notifications"
+msgstr "删除通知"
+
+#: bookwyrm/templates/notifications.html:51
+#, python-format
+msgid "favorited your review of %(book_title)s "
+msgstr "喜欢了你 对 %(book_title)s 的书评 "
+
+#: bookwyrm/templates/notifications.html:53
+#, python-format
+msgid "favorited your comment on %(book_title)s "
+msgstr "喜欢了你 对 %(book_title)s 的评论 "
+
+#: bookwyrm/templates/notifications.html:55
+#, python-format
+msgid "favorited your quote from %(book_title)s "
+msgstr "喜欢了你 来自 %(book_title)s 的引用 "
+
+#: bookwyrm/templates/notifications.html:57
+#, python-format
+msgid "favorited your status "
+msgstr "喜欢了你的 状态 "
+
+#: bookwyrm/templates/notifications.html:62
+#, python-format
+msgid "mentioned you in a review of %(book_title)s "
+msgstr "在 对 %(book_title)s 的书评 里提到了你"
+
+#: bookwyrm/templates/notifications.html:64
+#, python-format
+msgid "mentioned you in a comment on %(book_title)s "
+msgstr "在 对 %(book_title)s 的评论 里提到了你"
+
+#: bookwyrm/templates/notifications.html:66
+#, python-format
+msgid "mentioned you in a quote from %(book_title)s "
+msgstr "在 对 %(book_title)s 的引用 中提到了你"
+
+#: bookwyrm/templates/notifications.html:68
+#, python-format
+msgid "mentioned you in a status "
+msgstr "在 状态 中提到了你"
+
+#: bookwyrm/templates/notifications.html:73
+#, python-format
+msgid "replied to your review of %(book_title)s "
+msgstr "回复 了你的 对 %(book_title)s 的书评 "
+
+#: bookwyrm/templates/notifications.html:75
+#, python-format
+msgid "replied to your comment on %(book_title)s "
+msgstr "回复 了你的 对 %(book_title)s 的评论 "
+
+#: bookwyrm/templates/notifications.html:77
+#, python-format
+msgid "replied to your quote from %(book_title)s "
+msgstr "回复 了你 对 %(book_title)s 中的引用 "
+
+#: bookwyrm/templates/notifications.html:79
+#, python-format
+msgid "replied to your status "
+msgstr "回复 了你的 状态 "
+
+#: bookwyrm/templates/notifications.html:83
+msgid "followed you"
+msgstr "关注了你"
+
+#: bookwyrm/templates/notifications.html:86
+msgid "sent you a follow request"
+msgstr "向你发送了关注请求"
+
+#: bookwyrm/templates/notifications.html:92
+#, python-format
+msgid "boosted your review of %(book_title)s "
+msgstr "转发了你的 对 %(book_title)s 的书评 "
+
+#: bookwyrm/templates/notifications.html:94
+#, python-format
+msgid "boosted your comment on%(book_title)s "
+msgstr "转发了你的 对 %(book_title)s 的评论 "
+
+#: bookwyrm/templates/notifications.html:96
+#, python-format
+msgid "boosted your quote from %(book_title)s "
+msgstr "转发了你的 对 %(book_title)s 的引用 "
+
+#: bookwyrm/templates/notifications.html:98
+#, python-format
+msgid "boosted your status "
+msgstr "转发了你的 状态 "
+
+#: bookwyrm/templates/notifications.html:102
+#, python-format
+msgid " added %(book_title)s to your list \"%(list_name)s \""
+msgstr " 添加了 %(book_title)s 到你的列表 \"%(list_name)s \""
+
+#: bookwyrm/templates/notifications.html:104
+#, python-format
+msgid " suggested adding %(book_title)s to your list \"%(list_name)s \""
+msgstr " 推荐添加 %(book_title)s 到你的列表 \"%(list_name)s \""
+
+#: bookwyrm/templates/notifications.html:108
+#, python-format
+msgid " your import completed."
+msgstr " 你的 导入 已完成。"
+
+#: bookwyrm/templates/notifications.html:142
+msgid "You're all caught up!"
+msgstr "你什么也没错过!"
+
+#: bookwyrm/templates/password_reset.html:4
+#: bookwyrm/templates/password_reset.html:10
+#: bookwyrm/templates/password_reset_request.html:4
+#: bookwyrm/templates/password_reset_request.html:10
+msgid "Reset Password"
+msgstr "重设密码"
+
+#: bookwyrm/templates/password_reset.html:23
+#: bookwyrm/templates/preferences/change_password.html:18
+msgid "Confirm password:"
+msgstr "确认密码:"
+
+#: bookwyrm/templates/password_reset.html:30
+msgid "Confirm"
+msgstr "确认"
+
+#: bookwyrm/templates/password_reset_request.html:12
+msgid "A link to reset your password will be sent to your email address"
+msgstr "重设你的密码的链接将会被发送到你的邮箱地址"
+
+#: bookwyrm/templates/password_reset_request.html:16
+#: bookwyrm/templates/preferences/edit_user.html:38
+#: bookwyrm/templates/snippets/register_form.html:13
+msgid "Email address:"
+msgstr "邮箱地址:"
+
+#: bookwyrm/templates/password_reset_request.html:23
+msgid "Reset password"
+msgstr "重设密码"
+
+#: bookwyrm/templates/preferences/blocks.html:4
+#: bookwyrm/templates/preferences/blocks.html:7
+#: bookwyrm/templates/preferences/preferences_layout.html:23
+msgid "Blocked Users"
+msgstr "屏蔽的用户"
+
+#: bookwyrm/templates/preferences/blocks.html:12
+msgid "No users currently blocked."
+msgstr "当前没有被屏蔽的用户。"
+
+#: bookwyrm/templates/preferences/change_password.html:4
+#: bookwyrm/templates/preferences/change_password.html:7
+#: bookwyrm/templates/preferences/change_password.html:21
+#: bookwyrm/templates/preferences/preferences_layout.html:17
+msgid "Change Password"
+msgstr "更改密码"
+
+#: bookwyrm/templates/preferences/change_password.html:14
+msgid "New password:"
+msgstr "新密码:"
+
+#: bookwyrm/templates/preferences/edit_user.html:4
+#: bookwyrm/templates/preferences/edit_user.html:7
+msgid "Edit Profile"
+msgstr "编辑个人资料"
+
+#: bookwyrm/templates/preferences/edit_user.html:17
+msgid "Avatar:"
+msgstr "头像:"
+
+#: bookwyrm/templates/preferences/edit_user.html:24
+msgid "Display name:"
+msgstr "显示名称:"
+
+#: bookwyrm/templates/preferences/edit_user.html:31
+msgid "Summary:"
+msgstr "概要:"
+
+#: bookwyrm/templates/preferences/edit_user.html:46
+msgid "Manually approve followers:"
+msgstr "手动批准关注者:"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:11
+msgid "Account"
+msgstr "帐号"
+
+#: bookwyrm/templates/preferences/preferences_layout.html:20
+msgid "Relationships"
+msgstr "关系"
+
+#: bookwyrm/templates/search_results.html:33
+msgid "Didn't find what you were looking for?"
+msgstr "没有找到你想找的?"
+
+#: bookwyrm/templates/search_results.html:35
+msgid "Show results from other catalogues"
+msgstr "显示其它类别的结果"
+
+#: bookwyrm/templates/search_results.html:57
+msgid "Import book"
+msgstr "导入书目"
+
+#: bookwyrm/templates/search_results.html:67
+msgid "Hide results from other catalogues"
+msgstr "隐藏其它类别的结果"
+
+#: bookwyrm/templates/search_results.html:75
+msgid "Matching Users"
+msgstr "匹配的用户"
+
+#: bookwyrm/templates/search_results.html:77
+#, python-format
+msgid "No users found for \"%(query)s\""
+msgstr "没有找到 \"%(query)s\" 的用户"
+
+#: bookwyrm/templates/search_results.html:94
+#, python-format
+msgid "No lists found for \"%(query)s\""
+msgstr "没有找到 \"%(query)s\" 的条目"
+
+#: bookwyrm/templates/settings/admin_layout.html:4
+msgid "Administration"
+msgstr "管理"
+
+#: bookwyrm/templates/settings/admin_layout.html:15
+msgid "Manage Users"
+msgstr "管理用户"
+
+#: bookwyrm/templates/settings/admin_layout.html:23
+#: bookwyrm/templates/settings/federation.html:4
+msgid "Federated Servers"
+msgstr "互联的服务器"
+
+#: bookwyrm/templates/settings/admin_layout.html:28
+msgid "Instance Settings"
+msgstr "实例设置"
+
+#: bookwyrm/templates/settings/admin_layout.html:32
+#: bookwyrm/templates/settings/site.html:4
+#: bookwyrm/templates/settings/site.html:6
+msgid "Site Settings"
+msgstr "站点设置"
+
+#: bookwyrm/templates/settings/admin_layout.html:35
+#: bookwyrm/templates/settings/site.html:13
+msgid "Instance Info"
+msgstr "实例信息"
+
+#: bookwyrm/templates/settings/admin_layout.html:36
+#: bookwyrm/templates/settings/site.html:39
+msgid "Images"
+msgstr "图像"
+
+#: bookwyrm/templates/settings/admin_layout.html:37
+#: bookwyrm/templates/settings/site.html:59
+msgid "Footer Content"
+msgstr "页脚内容"
+
+#: bookwyrm/templates/settings/admin_layout.html:38
+#: bookwyrm/templates/settings/site.html:77
+msgid "Registration"
+msgstr "注册"
+
+#: bookwyrm/templates/settings/federation.html:10
+msgid "Server name"
+msgstr "服务器名称"
+
+#: bookwyrm/templates/settings/federation.html:11
+msgid "Software"
+msgstr "软件"
+
+#: bookwyrm/templates/settings/federation.html:12
+msgid "Status"
+msgstr "状态"
+
+#: bookwyrm/templates/settings/manage_invites.html:7
+msgid "Generate New Invite"
+msgstr "生成新的邀请"
+
+#: bookwyrm/templates/settings/manage_invites.html:13
+msgid "Expiry:"
+msgstr "过期:"
+
+#: bookwyrm/templates/settings/manage_invites.html:19
+msgid "Use limit:"
+msgstr "使用限制:"
+
+#: bookwyrm/templates/settings/manage_invites.html:26
+msgid "Create Invite"
+msgstr "创建邀请"
+
+#: bookwyrm/templates/settings/manage_invites.html:33
+msgid "Link"
+msgstr "链接"
+
+#: bookwyrm/templates/settings/manage_invites.html:34
+msgid "Expires"
+msgstr "过期"
+
+#: bookwyrm/templates/settings/manage_invites.html:35
+msgid "Max uses"
+msgstr "最大使用次数"
+
+#: bookwyrm/templates/settings/manage_invites.html:36
+msgid "Times used"
+msgstr "已使用次数"
+
+#: bookwyrm/templates/settings/manage_invites.html:39
+msgid "No active invites"
+msgstr "无有效的邀请"
+
+#: bookwyrm/templates/settings/site.html:15
+msgid "Instance Name:"
+msgstr "实例名称"
+
+#: bookwyrm/templates/settings/site.html:19
+msgid "Tagline:"
+msgstr "标语"
+
+#: bookwyrm/templates/settings/site.html:23
+msgid "Instance description:"
+msgstr "实例描述:"
+
+#: bookwyrm/templates/settings/site.html:27
+msgid "Code of conduct:"
+msgstr "行为准则:"
+
+#: bookwyrm/templates/settings/site.html:31
+msgid "Privacy Policy:"
+msgstr "隐私政策:"
+
+#: bookwyrm/templates/settings/site.html:42
+msgid "Logo:"
+msgstr "图标:"
+
+#: bookwyrm/templates/settings/site.html:46
+msgid "Logo small:"
+msgstr "小号图标:"
+
+#: bookwyrm/templates/settings/site.html:50
+msgid "Favicon:"
+msgstr "Favicon:"
+
+#: bookwyrm/templates/settings/site.html:61
+msgid "Support link:"
+msgstr "支持链接:"
+
+#: bookwyrm/templates/settings/site.html:65
+msgid "Support title:"
+msgstr "支持标题:"
+
+#: bookwyrm/templates/settings/site.html:69
+msgid "Admin email:"
+msgstr "管理员邮件:"
+
+#: bookwyrm/templates/settings/site.html:79
+msgid "Allow registration:"
+msgstr "允许注册:"
+
+#: bookwyrm/templates/settings/site.html:83
+msgid "Registration closed text:"
+msgstr "注册关闭文字:"
+
+#: bookwyrm/templates/snippets/block_button.html:5
+msgid "Block"
+msgstr "屏蔽"
+
+#: bookwyrm/templates/snippets/block_button.html:10
+msgid "Un-block"
+msgstr "取消屏蔽"
+
+#: bookwyrm/templates/snippets/book_titleby.html:3
+#, python-format
+msgid "%(title)s by "
+msgstr "%(title)s 来自"
+
+#: bookwyrm/templates/snippets/boost_button.html:8
+#: bookwyrm/templates/snippets/boost_button.html:9
+#: bookwyrm/templates/snippets/status/status_body.html:41
+#: bookwyrm/templates/snippets/status/status_body.html:42
+msgid "Boost status"
+msgstr "转发状态"
+
+#: bookwyrm/templates/snippets/boost_button.html:16
+#: bookwyrm/templates/snippets/boost_button.html:17
+msgid "Un-boost status"
+msgstr "取消转发状态"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:3
+msgid "Spoiler alert:"
+msgstr "剧透警告:"
+
+#: bookwyrm/templates/snippets/content_warning_field.html:4
+msgid "Spoilers ahead!"
+msgstr "前有剧透!"
+
+#: bookwyrm/templates/snippets/create_status.html:9
+msgid "Review"
+msgstr "书评"
+
+#: bookwyrm/templates/snippets/create_status.html:12
+#: bookwyrm/templates/snippets/create_status_form.html:44
+msgid "Comment"
+msgstr "评论"
+
+#: bookwyrm/templates/snippets/create_status.html:15
+msgid "Quote"
+msgstr "引用"
+
+#: bookwyrm/templates/snippets/create_status_form.html:21
+#: bookwyrm/templates/snippets/shelf.html:17
+msgid "Rating"
+msgstr "评价"
+
+#: bookwyrm/templates/snippets/create_status_form.html:23
+#: bookwyrm/templates/snippets/rate_action.html:14
+#: bookwyrm/templates/snippets/stars.html:3
+msgid "No rating"
+msgstr "没有评价"
+
+#: bookwyrm/templates/snippets/create_status_form.html:54
+msgid "Include spoiler alert"
+msgstr "加入剧透警告"
+
+#: bookwyrm/templates/snippets/create_status_form.html:60
+#: bookwyrm/templates/snippets/privacy-icons.html:15
+#: bookwyrm/templates/snippets/privacy-icons.html:16
+#: bookwyrm/templates/snippets/privacy_select.html:19
+msgid "Private"
+msgstr "私密"
+
+#: bookwyrm/templates/snippets/create_status_form.html:67
+msgid "Post"
+msgstr "发布"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:4
+msgid "Delete these read dates?"
+msgstr "删除这些阅读日期吗?"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:7
+#, python-format
+msgid "You are deleting this readthrough and its %(count)s associated progress updates."
+msgstr "你正要删除这篇阅读经过以及与之相关的 %(count)s 次进度更新。"
+
+#: bookwyrm/templates/snippets/delete_readthrough_modal.html:15
+#: bookwyrm/templates/snippets/follow_request_buttons.html:13
+msgid "Delete"
+msgstr "删除"
+
+#: bookwyrm/templates/snippets/fav_button.html:7
+#: bookwyrm/templates/snippets/fav_button.html:8
+#: bookwyrm/templates/snippets/status/status_body.html:45
+#: bookwyrm/templates/snippets/status/status_body.html:46
+msgid "Like status"
+msgstr "喜欢状态"
+
+#: bookwyrm/templates/snippets/fav_button.html:15
+#: bookwyrm/templates/snippets/fav_button.html:16
+msgid "Un-like status"
+msgstr "取消喜欢状态"
+
+#: bookwyrm/templates/snippets/follow_button.html:6
+msgid "Follow request already sent."
+msgstr "已经发送关注请求。"
+
+#: bookwyrm/templates/snippets/follow_button.html:19
+msgid "Send follow request"
+msgstr "发送关注请求"
+
+#: bookwyrm/templates/snippets/follow_button.html:21
+msgid "Follow"
+msgstr "关注"
+
+#: bookwyrm/templates/snippets/follow_button.html:27
+msgid "Unfollow"
+msgstr "取消关注"
+
+#: bookwyrm/templates/snippets/follow_request_buttons.html:8
+msgid "Accept"
+msgstr "接受"
+
+#: bookwyrm/templates/snippets/generated_status/goal.html:1
+#, python-format
+msgid "set a goal to read %(counter)s book in %(year)s"
+msgid_plural "set a goal to read %(counter)s books in %(year)s"
+msgstr[0] "设定了在 %(year)s 内要读 %(counter)s 书的目标"
+
+#: bookwyrm/templates/snippets/goal_card.html:21
+msgid "Dismiss message"
+msgstr "遣散消息"
+
+#: bookwyrm/templates/snippets/goal_card.html:22
+#, python-format
+msgid "You can set or change your reading goal any time from your profile page "
+msgstr "你可以在任何时候从你的个人资料页面 中设置或改变你的阅读目标"
+
+#: bookwyrm/templates/snippets/goal_form.html:9
+msgid "Reading goal:"
+msgstr "阅读目标:"
+
+#: bookwyrm/templates/snippets/goal_form.html:14
+msgid "books"
+msgstr "书目"
+
+#: bookwyrm/templates/snippets/goal_form.html:19
+msgid "Goal privacy:"
+msgstr "目标隐私:"
+
+#: bookwyrm/templates/snippets/goal_form.html:26
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:37
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:29
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:20
+msgid "Post to feed"
+msgstr "发布到消息流中"
+
+#: bookwyrm/templates/snippets/goal_form.html:30
+msgid "Set goal"
+msgstr "设置目标"
+
+#: bookwyrm/templates/snippets/goal_progress.html:5
+msgid "Success!"
+msgstr "成功!"
+
+#: bookwyrm/templates/snippets/goal_progress.html:7
+#, python-format
+msgid "%(percent)s%% complete!"
+msgstr "完成了 %(percent)s%% !"
+
+#: bookwyrm/templates/snippets/goal_progress.html:10
+#, python-format
+msgid "You've read %(read_count)s of %(goal_count)s books ."
+msgstr "你已经阅读了 %(goal_count)s 本书中的 %(read_count)s 本 。"
+
+#: bookwyrm/templates/snippets/goal_progress.html:12
+#, python-format
+msgid "%(username)s has read %(read_count)s of %(goal_count)s books ."
+msgstr "%(username)s 已经阅读了 %(goal_count)s 本书中的 %(read_count)s 本 。"
+
+#: bookwyrm/templates/snippets/pagination.html:7
+msgid "Previous"
+msgstr "往前"
+
+#: bookwyrm/templates/snippets/pagination.html:15
+msgid "Next"
+msgstr "往后"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:3
+#: bookwyrm/templates/snippets/privacy-icons.html:4
+#: bookwyrm/templates/snippets/privacy_select.html:10
+msgid "Public"
+msgstr "公开"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:7
+#: bookwyrm/templates/snippets/privacy-icons.html:8
+#: bookwyrm/templates/snippets/privacy_select.html:13
+msgid "Unlisted"
+msgstr "不公开"
+
+#: bookwyrm/templates/snippets/privacy-icons.html:12
+msgid "Followers-only"
+msgstr "仅关注者"
+
+#: bookwyrm/templates/snippets/privacy_select.html:6
+msgid "Post privacy"
+msgstr "发文隐私"
+
+#: bookwyrm/templates/snippets/privacy_select.html:16
+#: bookwyrm/templates/user/followers.html:13
+msgid "Followers"
+msgstr "关注者"
+
+#: bookwyrm/templates/snippets/progress_update.html:6
+msgid "Progress:"
+msgstr "进度:"
+
+#: bookwyrm/templates/snippets/progress_update.html:16
+#: bookwyrm/templates/snippets/readthrough_form.html:22
+msgid "pages"
+msgstr "页数"
+
+#: bookwyrm/templates/snippets/progress_update.html:17
+#: bookwyrm/templates/snippets/readthrough_form.html:23
+msgid "percent"
+msgstr "百分比"
+
+#: bookwyrm/templates/snippets/progress_update.html:25
+#, python-format
+msgid "of %(book.pages)s pages"
+msgstr "全书 %(book.pages)s 页"
+
+#: bookwyrm/templates/snippets/rate_action.html:4
+msgid "Leave a rating"
+msgstr "留下评价"
+
+#: bookwyrm/templates/snippets/rate_action.html:29
+msgid "Rate"
+msgstr "评价"
+
+#: bookwyrm/templates/snippets/readthrough.html:7
+msgid "Progress Updates:"
+msgstr "进度更新:"
+
+#: bookwyrm/templates/snippets/readthrough.html:11
+msgid "finished"
+msgstr "已完成"
+
+#: bookwyrm/templates/snippets/readthrough.html:14
+msgid "Show all updates"
+msgstr "显示所有更新"
+
+#: bookwyrm/templates/snippets/readthrough.html:30
+msgid "Delete this progress update"
+msgstr "删除此进度更新"
+
+#: bookwyrm/templates/snippets/readthrough.html:40
+msgid "started"
+msgstr "已开始"
+
+#: bookwyrm/templates/snippets/readthrough.html:46
+#: bookwyrm/templates/snippets/readthrough.html:60
+msgid "Edit read dates"
+msgstr "编辑阅读日期"
+
+#: bookwyrm/templates/snippets/readthrough.html:50
+msgid "Delete these read dates"
+msgstr "删除这些阅读日期"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:7
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:19
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:17
+msgid "Started reading"
+msgstr "已开始阅读"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:14
+msgid "Progress"
+msgstr "进度"
+
+#: bookwyrm/templates/snippets/readthrough_form.html:30
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:25
+msgid "Finished reading"
+msgstr "已完成阅读"
+
+#: bookwyrm/templates/snippets/register_form.html:32
+msgid "Sign Up"
+msgstr "注册"
+
+#: bookwyrm/templates/snippets/rss_title.html:5
+#: bookwyrm/templates/snippets/status/status_header.html:11
+msgid "rated"
+msgstr "评价了"
+
+#: bookwyrm/templates/snippets/rss_title.html:7
+#: bookwyrm/templates/snippets/status/status_header.html:13
+msgid "reviewed"
+msgstr "写了书评给"
+
+#: bookwyrm/templates/snippets/rss_title.html:9
+#: bookwyrm/templates/snippets/status/status_header.html:15
+msgid "commented on"
+msgstr "评论了"
+
+#: bookwyrm/templates/snippets/rss_title.html:11
+#: bookwyrm/templates/snippets/status/status_header.html:17
+msgid "quoted"
+msgstr "引用了"
+
+#: bookwyrm/templates/snippets/search_result_text.html:3
+#, python-format
+msgid "by %(author)s"
+msgstr "由 %(author)s 所著"
+
+#: bookwyrm/templates/snippets/shelf.html:12
+msgid "Published"
+msgstr "已出版"
+
+#: bookwyrm/templates/snippets/shelf.html:13
+msgid "Shelved"
+msgstr "已上架"
+
+#: bookwyrm/templates/snippets/shelf.html:14
+msgid "Started"
+msgstr "已开始"
+
+#: bookwyrm/templates/snippets/shelf.html:15
+msgid "Finished"
+msgstr "已完成"
+
+#: bookwyrm/templates/snippets/shelf.html:16
+msgid "External links"
+msgstr "外部链接"
+
+#: bookwyrm/templates/snippets/shelf.html:44
+msgid "OpenLibrary"
+msgstr "OpenLibrary"
+
+#: bookwyrm/templates/snippets/shelf.html:61
+msgid "This shelf is empty."
+msgstr "此书架是空的。"
+
+#: bookwyrm/templates/snippets/shelf.html:67
+msgid "Delete shelf"
+msgstr "删除书架"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:4
+msgid "Change shelf"
+msgstr "改变书架"
+
+#: bookwyrm/templates/snippets/shelf_selector.html:27
+msgid "Unshelve"
+msgstr "取下书架"
+
+#: bookwyrm/templates/snippets/shelve_button/finish_reading_modal.html:5
+#, python-format
+msgid "Finish \"%(book_title)s \""
+msgstr "完成 \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_dropdown.html:5
+msgid "More shelves"
+msgstr "更多书架"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:8
+msgid "Start reading"
+msgstr "开始阅读"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:13
+msgid "Finish reading"
+msgstr "完成阅读"
+
+#: bookwyrm/templates/snippets/shelve_button/shelve_button_options.html:16
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:26
+msgid "Want to read"
+msgstr "想要阅读"
+
+#: bookwyrm/templates/snippets/shelve_button/start_reading_modal.html:5
+#, python-format
+msgid "Start \"%(book_title)s \""
+msgstr "开始 \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/shelve_button/want_to_read_modal.html:5
+#, python-format
+msgid "Want to Read \"%(book_title)s \""
+msgstr "想要阅读 \"%(book_title)s \""
+
+#: bookwyrm/templates/snippets/status/status.html:9
+msgid "boosted"
+msgstr "转发了"
+
+#: bookwyrm/templates/snippets/status/status_body.html:24
+#: bookwyrm/templates/snippets/status/status_body.html:37
+#: bookwyrm/templates/snippets/status/status_body.html:38
+msgid "Reply"
+msgstr "回复"
+
+#: bookwyrm/templates/snippets/status/status_content.html:18
+#: bookwyrm/templates/snippets/trimmed_text.html:15
+msgid "Show more"
+msgstr "显示更多"
+
+#: bookwyrm/templates/snippets/status/status_content.html:25
+#: bookwyrm/templates/snippets/trimmed_text.html:25
+msgid "Show less"
+msgstr "显示更少"
+
+#: bookwyrm/templates/snippets/status/status_content.html:46
+msgid "Open image in new window"
+msgstr "在新窗口中打开图像"
+
+#: bookwyrm/templates/snippets/status/status_header.html:22
+#, fuzzy, python-format
+#| msgid "Direct Messages with %(username)s "
+msgid "replied to %(username)s's review "
+msgstr "与 %(username)s 私信"
+
+#: bookwyrm/templates/snippets/status/status_header.html:24
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's comment "
+msgstr "回复 了你的 状态 "
+
+#: bookwyrm/templates/snippets/status/status_header.html:26
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's quote "
+msgstr "回复 了你的 状态 "
+
+#: bookwyrm/templates/snippets/status/status_header.html:28
+#, fuzzy, python-format
+#| msgid "replied to your status "
+msgid "replied to %(username)s's status "
+msgstr "回复 了你的 状态 "
+
+#: bookwyrm/templates/snippets/status/status_options.html:7
+#: bookwyrm/templates/snippets/user_options.html:7
+msgid "More options"
+msgstr "更多选项"
+
+#: bookwyrm/templates/snippets/status/status_options.html:17
+msgid "Delete status"
+msgstr "删除发文"
+
+#: bookwyrm/templates/snippets/status/status_options.html:23
+#: bookwyrm/templates/snippets/user_options.html:13
+msgid "Send direct message"
+msgstr "发送私信"
+
+#: bookwyrm/templates/snippets/switch_edition_button.html:5
+msgid "Switch to this edition"
+msgstr "切换到此版本"
+
+#: bookwyrm/templates/snippets/tag.html:14
+msgid "Remove tag"
+msgstr "移除标签"
+
+#: bookwyrm/templates/tag.html:9
+#, python-format
+msgid "Books tagged \"%(tag.name)s\""
+msgstr "标有 \"%(tag.name)s\" 标签的书"
+
+#: bookwyrm/templates/user/create_shelf_form.html:5
+#: bookwyrm/templates/user/create_shelf_form.html:22
+msgid "Create Shelf"
+msgstr "创建书架"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:5
+msgid "Edit Shelf"
+msgstr "编辑书架"
+
+#: bookwyrm/templates/user/edit_shelf_form.html:26
+msgid "Update shelf"
+msgstr "更新书架"
+
+#: bookwyrm/templates/user/followers.html:7
+#: bookwyrm/templates/user/following.html:7 bookwyrm/templates/user/user.html:9
+msgid "User Profile"
+msgstr "用户个人资料"
+
+#: bookwyrm/templates/user/followers.html:29
+#, python-format
+msgid "%(username)s has no followers"
+msgstr "%(username)s 没有关注者"
+
+#: bookwyrm/templates/user/following.html:13
+msgid "Following"
+msgstr "正在关注"
+
+#: bookwyrm/templates/user/following.html:29
+#, python-format
+msgid "%(username)s isn't following any users"
+msgstr "%(username)s 没有关注任何用户"
+
+#: bookwyrm/templates/user/lists.html:9
+msgid "Your Lists"
+msgstr "你的列表"
+
+#: bookwyrm/templates/user/lists.html:11
+#, python-format
+msgid "Lists: %(username)s"
+msgstr "列表: %(username)s"
+
+#: bookwyrm/templates/user/lists.html:17 bookwyrm/templates/user/lists.html:29
+msgid "Create list"
+msgstr "创建列表"
+
+#: bookwyrm/templates/user/shelf.html:9
+msgid "Your Shelves"
+msgstr "你的书架"
+
+#: bookwyrm/templates/user/shelf.html:11
+#, python-format
+msgid "%(username)s: Shelves"
+msgstr "%(username)s: 书架"
+
+#: bookwyrm/templates/user/shelf.html:33
+msgid "Create shelf"
+msgstr "创建书架"
+
+#: bookwyrm/templates/user/shelf.html:54
+msgid "Edit shelf"
+msgstr "编辑书架"
+
+#: bookwyrm/templates/user/user.html:15
+msgid "Edit profile"
+msgstr "编辑个人资料"
+
+#: bookwyrm/templates/user/user.html:26
+#: bookwyrm/templates/user/user_layout.html:68
+msgid "Shelves"
+msgstr "书架"
+
+#: bookwyrm/templates/user/user.html:31
+#, python-format
+msgid "See all %(size)s"
+msgstr "查看所有 %(size)s"
+
+#: bookwyrm/templates/user/user.html:44
+#, python-format
+msgid "See all %(shelf_count)s shelves"
+msgstr "查看所有 %(shelf_count)s 个书架"
+
+#: bookwyrm/templates/user/user.html:56
+#, python-format
+msgid "Set a reading goal for %(year)s"
+msgstr "设定 %(year)s 的阅读目标"
+
+#: bookwyrm/templates/user/user.html:62
+msgid "User Activity"
+msgstr "用户活动"
+
+#: bookwyrm/templates/user/user.html:65
+msgid "RSS feed"
+msgstr "RSS 流"
+
+#: bookwyrm/templates/user/user.html:76
+msgid "No activities yet!"
+msgstr "还没有活动!"
+
+#: bookwyrm/templates/user/user_layout.html:32
+msgid "Follow Requests"
+msgstr "关注请求"
+
+#: bookwyrm/templates/user/user_layout.html:50
+msgid "Activity"
+msgstr "活动"
+
+#: bookwyrm/templates/user/user_layout.html:56
+msgid "Reading Goal"
+msgstr "阅读目标"
+
+#: bookwyrm/templates/user/user_preview.html:13
+#, python-format
+msgid "Joined %(date)s"
+msgstr "已加入 %(date)s"
+
+#: bookwyrm/templates/user/user_preview.html:15
+#, python-format
+msgid "%(counter)s follower"
+msgid_plural "%(counter)s followers"
+msgstr[0] "%(counter)s 个关注者"
+
+#: bookwyrm/templates/user/user_preview.html:16
+#, python-format
+msgid "%(counter)s following"
+msgstr "关注着 %(counter)s 人"
+
+#~ msgid "Created and curated by"
+#~ msgstr "创建者及策展者为"
+
+#~ msgid "Created by"
+#~ msgstr "创建者为"
+
+#~ msgid "Create New Shelf"
+#~ msgstr "新建书架"
+
+#~ msgid "Create new list"
+#~ msgstr "新建列表"
+
+#~ msgid "Added by"
+#~ msgstr "添加来自"
diff --git a/requirements.txt b/requirements.txt
index e5d7798d..f354fd43 100644
--- a/requirements.txt
+++ b/requirements.txt
@@ -1,5 +1,4 @@
celery==4.4.2
-coverage==5.1
Django==3.0.7
django-model-utils==4.0.0
environs==7.2.0
@@ -8,11 +7,15 @@ Markdown==3.3.3
Pillow>=7.1.0
psycopg2==2.8.4
pycryptodome==3.9.4
-pytest-django==4.1.0
-pytest==6.1.2
-pytest-cov==2.10.1
python-dateutil==2.8.1
redis==3.4.1
requests==2.22.0
responses==0.10.14
django-rename-app==0.1.2
+
+# Dev
+black==20.8b1
+coverage==5.1
+pytest-django==4.1.0
+pytest==6.1.2
+pytest-cov==2.10.1