mirror of
https://github.com/bookwyrm-social/bookwyrm.git
synced 2025-01-22 06:58:07 +00:00
Merge branch 'main' into production
This commit is contained in:
commit
af92e3e9a4
17 changed files with 172 additions and 117 deletions
|
@ -1,14 +0,0 @@
|
||||||
""" handle reading a csv from goodreads """
|
|
||||||
from bookwyrm.importer import Importer
|
|
||||||
|
|
||||||
# GoodReads is the default importer, thus Importer follows its structure. For a more complete example of overriding see librarything_import.py
|
|
||||||
|
|
||||||
|
|
||||||
class GoodreadsImporter(Importer):
|
|
||||||
service = "GoodReads"
|
|
||||||
|
|
||||||
def parse_fields(self, data):
|
|
||||||
data.update({"import_source": self.service})
|
|
||||||
# add missing 'Date Started' field
|
|
||||||
data.update({"Date Started": None})
|
|
||||||
return data
|
|
5
bookwyrm/importers/__init__.py
Normal file
5
bookwyrm/importers/__init__.py
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
""" import classes """
|
||||||
|
|
||||||
|
from .importer import Importer
|
||||||
|
from .goodreads_import import GoodreadsImporter
|
||||||
|
from .librarything_import import LibrarythingImporter
|
16
bookwyrm/importers/goodreads_import.py
Normal file
16
bookwyrm/importers/goodreads_import.py
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
""" handle reading a csv from goodreads """
|
||||||
|
from . import Importer
|
||||||
|
|
||||||
|
|
||||||
|
class GoodreadsImporter(Importer):
|
||||||
|
"""GoodReads is the default importer, thus Importer follows its structure.
|
||||||
|
For a more complete example of overriding see librarything_import.py"""
|
||||||
|
|
||||||
|
service = "GoodReads"
|
||||||
|
|
||||||
|
def parse_fields(self, entry):
|
||||||
|
""" handle the specific fields in goodreads csvs """
|
||||||
|
entry.update({"import_source": self.service})
|
||||||
|
# add missing 'Date Started' field
|
||||||
|
entry.update({"Date Started": None})
|
||||||
|
return entry
|
|
@ -10,6 +10,8 @@ logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class Importer:
|
class Importer:
|
||||||
|
""" Generic class for csv data import from an outside service """
|
||||||
|
|
||||||
service = "Unknown"
|
service = "Unknown"
|
||||||
delimiter = ","
|
delimiter = ","
|
||||||
encoding = "UTF-8"
|
encoding = "UTF-8"
|
||||||
|
@ -29,10 +31,12 @@ class Importer:
|
||||||
self.save_item(job, index, entry)
|
self.save_item(job, index, entry)
|
||||||
return job
|
return job
|
||||||
|
|
||||||
def save_item(self, job, index, data):
|
def save_item(self, job, index, data): # pylint: disable=no-self-use
|
||||||
|
""" creates and saves an import item """
|
||||||
ImportItem(job=job, index=index, data=data).save()
|
ImportItem(job=job, index=index, data=data).save()
|
||||||
|
|
||||||
def parse_fields(self, entry):
|
def parse_fields(self, entry):
|
||||||
|
""" updates csv data with additional info """
|
||||||
entry.update({"import_source": self.service})
|
entry.update({"import_source": self.service})
|
||||||
return entry
|
return entry
|
||||||
|
|
|
@ -1,35 +1,35 @@
|
||||||
""" handle reading a csv from librarything """
|
""" handle reading a csv from librarything """
|
||||||
import csv
|
|
||||||
import re
|
import re
|
||||||
import math
|
import math
|
||||||
|
|
||||||
from bookwyrm import models
|
from . import Importer
|
||||||
from bookwyrm.models import ImportItem
|
|
||||||
from bookwyrm.importer import Importer
|
|
||||||
|
|
||||||
|
|
||||||
class LibrarythingImporter(Importer):
|
class LibrarythingImporter(Importer):
|
||||||
|
""" csv downloads from librarything """
|
||||||
|
|
||||||
service = "LibraryThing"
|
service = "LibraryThing"
|
||||||
delimiter = "\t"
|
delimiter = "\t"
|
||||||
encoding = "ISO-8859-1"
|
encoding = "ISO-8859-1"
|
||||||
# mandatory_fields : fields matching the book title and author
|
# mandatory_fields : fields matching the book title and author
|
||||||
mandatory_fields = ["Title", "Primary Author"]
|
mandatory_fields = ["Title", "Primary Author"]
|
||||||
|
|
||||||
def parse_fields(self, initial):
|
def parse_fields(self, entry):
|
||||||
|
""" custom parsing for librarything """
|
||||||
data = {}
|
data = {}
|
||||||
data["import_source"] = self.service
|
data["import_source"] = self.service
|
||||||
data["Book Id"] = initial["Book Id"]
|
data["Book Id"] = entry["Book Id"]
|
||||||
data["Title"] = initial["Title"]
|
data["Title"] = entry["Title"]
|
||||||
data["Author"] = initial["Primary Author"]
|
data["Author"] = entry["Primary Author"]
|
||||||
data["ISBN13"] = initial["ISBN"]
|
data["ISBN13"] = entry["ISBN"]
|
||||||
data["My Review"] = initial["Review"]
|
data["My Review"] = entry["Review"]
|
||||||
if initial["Rating"]:
|
if entry["Rating"]:
|
||||||
data["My Rating"] = math.ceil(float(initial["Rating"]))
|
data["My Rating"] = math.ceil(float(entry["Rating"]))
|
||||||
else:
|
else:
|
||||||
data["My Rating"] = ""
|
data["My Rating"] = ""
|
||||||
data["Date Added"] = re.sub("\[|\]", "", initial["Entry Date"])
|
data["Date Added"] = re.sub(r"\[|\]", "", entry["Entry Date"])
|
||||||
data["Date Started"] = re.sub("\[|\]", "", initial["Date Started"])
|
data["Date Started"] = re.sub(r"\[|\]", "", entry["Date Started"])
|
||||||
data["Date Read"] = re.sub("\[|\]", "", initial["Date Read"])
|
data["Date Read"] = re.sub(r"\[|\]", "", entry["Date Read"])
|
||||||
|
|
||||||
data["Exclusive Shelf"] = None
|
data["Exclusive Shelf"] = None
|
||||||
if data["Date Read"]:
|
if data["Date Read"]:
|
|
@ -1,6 +1,7 @@
|
||||||
""" make a list of books!! """
|
""" make a list of books!! """
|
||||||
from django.apps import apps
|
from django.apps import apps
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
from django.utils import timezone
|
||||||
|
|
||||||
from bookwyrm import activitypub
|
from bookwyrm import activitypub
|
||||||
from bookwyrm.settings import DOMAIN
|
from bookwyrm.settings import DOMAIN
|
||||||
|
@ -79,6 +80,10 @@ class ListItem(CollectionItemMixin, BookWyrmModel):
|
||||||
""" create a notification too """
|
""" create a notification too """
|
||||||
created = not bool(self.id)
|
created = not bool(self.id)
|
||||||
super().save(*args, **kwargs)
|
super().save(*args, **kwargs)
|
||||||
|
# tick the updated date on the parent list
|
||||||
|
self.book_list.updated_date = timezone.now()
|
||||||
|
self.book_list.save(broadcast=False)
|
||||||
|
|
||||||
list_owner = self.book_list.user
|
list_owner = self.book_list.user
|
||||||
# create a notification if somoene ELSE added to a local user's list
|
# create a notification if somoene ELSE added to a local user's list
|
||||||
if created and list_owner.local and list_owner != self.user:
|
if created and list_owner.local and list_owner != self.user:
|
||||||
|
|
|
@ -7,36 +7,43 @@
|
||||||
{% block content %}
|
{% block content %}
|
||||||
<div class="block">
|
<div class="block">
|
||||||
<h1 class="title">{% trans "Import Books" %}</h1>
|
<h1 class="title">{% trans "Import Books" %}</h1>
|
||||||
<form name="import" action="/import" method="post" enctype="multipart/form-data">
|
<form class="box" name="import" action="/import" method="post" enctype="multipart/form-data">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
|
|
||||||
<label class="label" for="source">
|
<div class="columns">
|
||||||
<p>{% trans "Data source" %}</p>
|
<div class="column is-half">
|
||||||
<div class="select {{ class }}">
|
<label class="label" for="source">
|
||||||
<select name="source" id="source">
|
{% trans "Data source:" %}
|
||||||
<option value="GoodReads" {% if current == 'GoodReads' %}selected{% endif %}>
|
</label>
|
||||||
GoodReads
|
<div class="select block">
|
||||||
</option>
|
<select name="source" id="source">
|
||||||
<option value="LibraryThing" {% if current == 'LibraryThing' %}selected{% endif %}>
|
<option value="GoodReads" {% if current == 'GoodReads' %}selected{% endif %}>
|
||||||
LibraryThing
|
GoodReads (CSV)
|
||||||
</option>
|
</option>
|
||||||
</select>
|
<option value="LibraryThing" {% if current == 'LibraryThing' %}selected{% endif %}>
|
||||||
|
LibraryThing (TSV)
|
||||||
|
</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label class="label" for="id_csv_field">{% trans "Data file:" %}</label>
|
||||||
|
{{ import_form.csv_file }}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</label>
|
|
||||||
|
|
||||||
<div class="field">
|
<div class="column is-half">
|
||||||
{{ import_form.as_p }}
|
<div class="field">
|
||||||
|
<label class="label">
|
||||||
|
<input type="checkbox" name="include_reviews" checked> {% trans "Include reviews" %}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="field">
|
||||||
|
<label class="label">
|
||||||
|
<p>{% trans "Privacy setting for imported reviews:" %}</p>
|
||||||
|
{% include 'snippets/privacy_select.html' with no_label=True %}
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="field">
|
|
||||||
<label class="label">
|
|
||||||
<input type="checkbox" name="include_reviews" checked> {% trans "Include reviews" %}
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
<div class="field">
|
|
||||||
<label class="label">
|
|
||||||
<p>{% trans "Privacy setting for imported reviews:" %}</p>
|
|
||||||
{% include 'snippets/privacy_select.html' with no_label=True %}
|
|
||||||
</label>
|
|
||||||
</div>
|
</div>
|
||||||
<button class="button is-primary" type="submit">{% trans "Import" %}</button>
|
<button class="button is-primary" type="submit">{% trans "Import" %}</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
|
@ -14,7 +14,13 @@
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-content is-flex-grow-0">
|
<div class="card-content is-flex-grow-0">
|
||||||
{% if list.description %}{{ list.description | to_markdown | safe | truncatewords_html:20 }}{% endif %}
|
<div {% if list.description %}title="{{ list.description }}"{% endif %}>
|
||||||
|
{% if list.description %}
|
||||||
|
{{ list.description|to_markdown|safe|truncatechars_html:30 }}
|
||||||
|
{% else %}
|
||||||
|
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
<p class="subtitle help">
|
<p class="subtitle help">
|
||||||
{% include 'lists/created_text.html' with list=list %}
|
{% include 'lists/created_text.html' with list=list %}
|
||||||
</p>
|
</p>
|
||||||
|
|
|
@ -1,17 +1,19 @@
|
||||||
{% extends 'layout.html' %}
|
{% extends 'layout.html' %}
|
||||||
|
{% load bookwyrm_tags %}
|
||||||
{% load i18n %}
|
{% load i18n %}
|
||||||
|
|
||||||
{% block title %}{% trans "Lists" %}{% endblock %}
|
{% block title %}{% trans "Lists" %}{% endblock %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
|
||||||
<header class="block">
|
|
||||||
<h1 class="title">{% trans "Lists" %}</h1>
|
|
||||||
</header>
|
|
||||||
{% if request.user.is_authenticated and not lists.has_previous %}
|
|
||||||
<header class="block columns is-mobile">
|
<header class="block columns is-mobile">
|
||||||
<div class="column">
|
<div class="column">
|
||||||
<h2 class="title">{% trans "Your lists" %}</h2>
|
<h1 class="title">
|
||||||
|
{% trans "Lists" %}
|
||||||
|
{% if request.user.is_authenticated %}
|
||||||
|
<a class="help has-text-weight-normal" href="{% url 'user-lists' request.user|username %}">Your lists</a>
|
||||||
|
{% endif %}
|
||||||
|
</h1>
|
||||||
</div>
|
</div>
|
||||||
<div class="column is-narrow">
|
<div class="column is-narrow">
|
||||||
{% trans "Create List" as button_text %}
|
{% trans "Create List" as button_text %}
|
||||||
|
@ -23,23 +25,11 @@
|
||||||
{% include 'lists/create_form.html' with controls_text="create-list" %}
|
{% include 'lists/create_form.html' with controls_text="create-list" %}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<section class="block content">
|
|
||||||
{% if request.user.list_set.exists %}
|
|
||||||
{% include 'lists/list_items.html' with lists=request.user.list_set.all|slice:4 %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
{% if request.user.list_set.count > 4 %}
|
|
||||||
<a href="{% url 'user-lists' request.user.localname %}">{% blocktrans with size=request.user.list_set.count %}See all {{ size }} lists{% endblocktrans %}</a>
|
|
||||||
{% endif %}
|
|
||||||
</section>
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
|
|
||||||
{% if lists %}
|
{% if lists %}
|
||||||
<section class="block content">
|
<section class="block content">
|
||||||
<h2 class="title">{% trans "Recent Lists" %}</h2>
|
|
||||||
{% include 'lists/list_items.html' with lists=lists %}
|
{% include 'lists/list_items.html' with lists=lists %}
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
{% include 'snippets/pagination.html' with page=lists path=path %}
|
{% include 'snippets/pagination.html' with page=lists path=path %}
|
||||||
</div>
|
</div>
|
||||||
|
|
1
bookwyrm/tests/importers/__init__.py
Normal file
1
bookwyrm/tests/importers/__init__.py
Normal file
|
@ -0,0 +1 @@
|
||||||
|
from . import *
|
|
@ -7,8 +7,9 @@ from unittest.mock import patch
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
import responses
|
import responses
|
||||||
|
|
||||||
from bookwyrm import models, importer
|
from bookwyrm import models
|
||||||
from bookwyrm.goodreads_import import GoodreadsImporter
|
from bookwyrm.importers import GoodreadsImporter
|
||||||
|
from bookwyrm.importers.importer import import_data, handle_imported_book
|
||||||
from bookwyrm.settings import DOMAIN
|
from bookwyrm.settings import DOMAIN
|
||||||
|
|
||||||
|
|
||||||
|
@ -18,7 +19,7 @@ class GoodreadsImport(TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
""" use a test csv """
|
""" use a test csv """
|
||||||
self.importer = GoodreadsImporter()
|
self.importer = GoodreadsImporter()
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
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(
|
self.user = models.User.objects.create_user(
|
||||||
"mouse", "mouse@mouse.mouse", "password", local=True
|
"mouse", "mouse@mouse.mouse", "password", local=True
|
||||||
|
@ -81,7 +82,7 @@ class GoodreadsImport(TestCase):
|
||||||
import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
|
import_job = self.importer.create_job(self.user, self.csv, False, "unlisted")
|
||||||
MockTask = namedtuple("Task", ("id"))
|
MockTask = namedtuple("Task", ("id"))
|
||||||
mock_task = MockTask(7)
|
mock_task = MockTask(7)
|
||||||
with patch("bookwyrm.importer.import_data.delay") as start:
|
with patch("bookwyrm.importers.importer.import_data.delay") as start:
|
||||||
start.return_value = mock_task
|
start.return_value = mock_task
|
||||||
self.importer.start_import(import_job)
|
self.importer.start_import(import_job)
|
||||||
import_job.refresh_from_db()
|
import_job.refresh_from_db()
|
||||||
|
@ -97,8 +98,8 @@ class GoodreadsImport(TestCase):
|
||||||
"bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
|
"bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
|
||||||
) as resolve:
|
) as resolve:
|
||||||
resolve.return_value = book
|
resolve.return_value = book
|
||||||
with patch("bookwyrm.importer.handle_imported_book"):
|
with patch("bookwyrm.importers.importer.handle_imported_book"):
|
||||||
importer.import_data(self.importer.service, import_job.id)
|
import_data(self.importer.service, import_job.id)
|
||||||
|
|
||||||
import_item = models.ImportItem.objects.get(job=import_job, index=0)
|
import_item = models.ImportItem.objects.get(job=import_job, index=0)
|
||||||
self.assertEqual(import_item.book.id, book.id)
|
self.assertEqual(import_item.book.id, book.id)
|
||||||
|
@ -109,7 +110,7 @@ class GoodreadsImport(TestCase):
|
||||||
self.assertIsNone(shelf.books.first())
|
self.assertIsNone(shelf.books.first())
|
||||||
|
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
csv_file = open(datafile, "r")
|
csv_file = open(datafile, "r")
|
||||||
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -119,7 +120,7 @@ class GoodreadsImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -143,7 +144,7 @@ class GoodreadsImport(TestCase):
|
||||||
models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
|
models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
|
||||||
|
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
csv_file = open(datafile, "r")
|
csv_file = open(datafile, "r")
|
||||||
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -153,7 +154,7 @@ class GoodreadsImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -173,7 +174,7 @@ class GoodreadsImport(TestCase):
|
||||||
""" re-importing books """
|
""" re-importing books """
|
||||||
shelf = self.user.shelf_set.filter(identifier="read").first()
|
shelf = self.user.shelf_set.filter(identifier="read").first()
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
csv_file = open(datafile, "r")
|
csv_file = open(datafile, "r")
|
||||||
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
for index, entry in enumerate(list(csv.DictReader(csv_file))):
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -183,10 +184,10 @@ class GoodreadsImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
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(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -207,7 +208,7 @@ class GoodreadsImport(TestCase):
|
||||||
def test_handle_imported_book_review(self, _):
|
def test_handle_imported_book_review(self, _):
|
||||||
""" goodreads review import """
|
""" goodreads review import """
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
csv_file = open(datafile, "r")
|
csv_file = open(datafile, "r")
|
||||||
entry = list(csv.DictReader(csv_file))[2]
|
entry = list(csv.DictReader(csv_file))[2]
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -216,7 +217,7 @@ class GoodreadsImport(TestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
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)
|
review = models.Review.objects.get(book=self.book, user=self.user)
|
||||||
|
@ -230,7 +231,7 @@ class GoodreadsImport(TestCase):
|
||||||
def test_handle_imported_book_reviews_disabled(self):
|
def test_handle_imported_book_reviews_disabled(self):
|
||||||
""" goodreads review import """
|
""" goodreads review import """
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/goodreads.csv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/goodreads.csv")
|
||||||
csv_file = open(datafile, "r")
|
csv_file = open(datafile, "r")
|
||||||
entry = list(csv.DictReader(csv_file))[2]
|
entry = list(csv.DictReader(csv_file))[2]
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -239,7 +240,7 @@ class GoodreadsImport(TestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "unlisted"
|
self.importer.service, self.user, import_item, False, "unlisted"
|
||||||
)
|
)
|
||||||
self.assertFalse(
|
self.assertFalse(
|
|
@ -6,8 +6,9 @@ from unittest.mock import patch
|
||||||
from django.test import TestCase
|
from django.test import TestCase
|
||||||
import responses
|
import responses
|
||||||
|
|
||||||
from bookwyrm import models, importer
|
from bookwyrm import models
|
||||||
from bookwyrm.librarything_import import LibrarythingImporter
|
from bookwyrm.importers import LibrarythingImporter
|
||||||
|
from bookwyrm.importers.importer import import_data, handle_imported_book
|
||||||
from bookwyrm.settings import DOMAIN
|
from bookwyrm.settings import DOMAIN
|
||||||
|
|
||||||
|
|
||||||
|
@ -17,7 +18,7 @@ class LibrarythingImport(TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
""" use a test tsv """
|
""" use a test tsv """
|
||||||
self.importer = LibrarythingImporter()
|
self.importer = LibrarythingImporter()
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
|
|
||||||
# Librarything generates latin encoded exports...
|
# Librarything generates latin encoded exports...
|
||||||
self.csv = open(datafile, "r", encoding=self.importer.encoding)
|
self.csv = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
|
@ -87,8 +88,8 @@ class LibrarythingImport(TestCase):
|
||||||
"bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
|
"bookwyrm.models.import_job.ImportItem.get_book_from_isbn"
|
||||||
) as resolve:
|
) as resolve:
|
||||||
resolve.return_value = book
|
resolve.return_value = book
|
||||||
with patch("bookwyrm.importer.handle_imported_book"):
|
with patch("bookwyrm.importers.importer.handle_imported_book"):
|
||||||
importer.import_data(self.importer.service, import_job.id)
|
import_data(self.importer.service, import_job.id)
|
||||||
|
|
||||||
import_item = models.ImportItem.objects.get(job=import_job, index=0)
|
import_item = models.ImportItem.objects.get(job=import_job, index=0)
|
||||||
self.assertEqual(import_item.book.id, book.id)
|
self.assertEqual(import_item.book.id, book.id)
|
||||||
|
@ -99,7 +100,7 @@ class LibrarythingImport(TestCase):
|
||||||
self.assertIsNone(shelf.books.first())
|
self.assertIsNone(shelf.books.first())
|
||||||
|
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
for index, entry in enumerate(
|
for index, entry in enumerate(
|
||||||
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
||||||
|
@ -111,7 +112,7 @@ class LibrarythingImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -135,7 +136,7 @@ class LibrarythingImport(TestCase):
|
||||||
models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
|
models.ShelfBook.objects.create(shelf=shelf, user=self.user, book=self.book)
|
||||||
|
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
for index, entry in enumerate(
|
for index, entry in enumerate(
|
||||||
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
||||||
|
@ -147,7 +148,7 @@ class LibrarythingImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -167,7 +168,7 @@ class LibrarythingImport(TestCase):
|
||||||
""" re-importing books """
|
""" re-importing books """
|
||||||
shelf = self.user.shelf_set.filter(identifier="read").first()
|
shelf = self.user.shelf_set.filter(identifier="read").first()
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
for index, entry in enumerate(
|
for index, entry in enumerate(
|
||||||
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))
|
||||||
|
@ -179,10 +180,10 @@ class LibrarythingImport(TestCase):
|
||||||
break
|
break
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
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(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "public"
|
self.importer.service, self.user, import_item, False, "public"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -203,7 +204,7 @@ class LibrarythingImport(TestCase):
|
||||||
def test_handle_imported_book_review(self, _):
|
def test_handle_imported_book_review(self, _):
|
||||||
""" librarything review import """
|
""" librarything review import """
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[0]
|
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[0]
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -212,7 +213,7 @@ class LibrarythingImport(TestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
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)
|
review = models.Review.objects.get(book=self.book, user=self.user)
|
||||||
|
@ -226,7 +227,7 @@ class LibrarythingImport(TestCase):
|
||||||
def test_handle_imported_book_reviews_disabled(self):
|
def test_handle_imported_book_reviews_disabled(self):
|
||||||
""" librarything review import """
|
""" librarything review import """
|
||||||
import_job = models.ImportJob.objects.create(user=self.user)
|
import_job = models.ImportJob.objects.create(user=self.user)
|
||||||
datafile = pathlib.Path(__file__).parent.joinpath("data/librarything.tsv")
|
datafile = pathlib.Path(__file__).parent.joinpath("../data/librarything.tsv")
|
||||||
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
csv_file = open(datafile, "r", encoding=self.importer.encoding)
|
||||||
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[2]
|
entry = list(csv.DictReader(csv_file, delimiter=self.importer.delimiter))[2]
|
||||||
entry = self.importer.parse_fields(entry)
|
entry = self.importer.parse_fields(entry)
|
||||||
|
@ -235,7 +236,7 @@ class LibrarythingImport(TestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
with patch("bookwyrm.models.activitypub_mixin.broadcast_task.delay"):
|
||||||
importer.handle_imported_book(
|
handle_imported_book(
|
||||||
self.importer.service, self.user, import_item, False, "unlisted"
|
self.importer.service, self.user, import_item, False, "unlisted"
|
||||||
)
|
)
|
||||||
self.assertFalse(
|
self.assertFalse(
|
|
@ -66,6 +66,18 @@ class FeedViews(TestCase):
|
||||||
self.assertIsInstance(result, ActivitypubResponse)
|
self.assertIsInstance(result, ActivitypubResponse)
|
||||||
self.assertEqual(result.status_code, 200)
|
self.assertEqual(result.status_code, 200)
|
||||||
|
|
||||||
|
def test_status_page_not_found(self, *_):
|
||||||
|
""" there are so many views, this just makes sure it LOADS """
|
||||||
|
view = views.Status.as_view()
|
||||||
|
|
||||||
|
request = self.factory.get("")
|
||||||
|
request.user = self.local_user
|
||||||
|
with patch("bookwyrm.views.feed.is_api_request") as is_api:
|
||||||
|
is_api.return_value = False
|
||||||
|
result = view(request, "mouse", 12345)
|
||||||
|
|
||||||
|
self.assertEqual(result.status_code, 404)
|
||||||
|
|
||||||
def test_status_page_with_image(self, *_):
|
def test_status_page_with_image(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()
|
view = views.Status.as_view()
|
||||||
|
|
|
@ -45,3 +45,22 @@ class ImportViews(TestCase):
|
||||||
self.assertIsInstance(result, TemplateResponse)
|
self.assertIsInstance(result, TemplateResponse)
|
||||||
result.render()
|
result.render()
|
||||||
self.assertEqual(result.status_code, 200)
|
self.assertEqual(result.status_code, 200)
|
||||||
|
|
||||||
|
def test_retry_import(self):
|
||||||
|
""" retry failed items """
|
||||||
|
view = views.ImportStatus.as_view()
|
||||||
|
import_job = models.ImportJob.objects.create(
|
||||||
|
user=self.local_user, privacy="unlisted"
|
||||||
|
)
|
||||||
|
request = self.factory.post("")
|
||||||
|
request.user = self.local_user
|
||||||
|
|
||||||
|
with patch("bookwyrm.importers.Importer.start_import"):
|
||||||
|
view(request, import_job.id)
|
||||||
|
|
||||||
|
self.assertEqual(models.ImportJob.objects.count(), 2)
|
||||||
|
retry_job = models.ImportJob.objects.last()
|
||||||
|
|
||||||
|
self.assertTrue(retry_job.retry)
|
||||||
|
self.assertEqual(retry_job.user, self.local_user)
|
||||||
|
self.assertEqual(retry_job.privacy, "unlisted")
|
||||||
|
|
|
@ -115,7 +115,7 @@ class Status(View):
|
||||||
status = models.Status.objects.select_subclasses().get(
|
status = models.Status.objects.select_subclasses().get(
|
||||||
id=status_id, deleted=False
|
id=status_id, deleted=False
|
||||||
)
|
)
|
||||||
except ValueError:
|
except (ValueError, models.Status.DoesNotExist):
|
||||||
return HttpResponseNotFound()
|
return HttpResponseNotFound()
|
||||||
|
|
||||||
# the url should have the poster's username in it
|
# the url should have the poster's username in it
|
||||||
|
|
|
@ -9,7 +9,8 @@ from django.template.response import TemplateResponse
|
||||||
from django.utils.decorators import method_decorator
|
from django.utils.decorators import method_decorator
|
||||||
from django.views import View
|
from django.views import View
|
||||||
|
|
||||||
from bookwyrm import forms, goodreads_import, librarything_import, models
|
from bookwyrm import forms, models
|
||||||
|
from bookwyrm.importers import Importer, LibrarythingImporter, GoodreadsImporter
|
||||||
from bookwyrm.tasks import app
|
from bookwyrm.tasks import app
|
||||||
|
|
||||||
# pylint: disable= no-self-use
|
# pylint: disable= no-self-use
|
||||||
|
@ -40,10 +41,10 @@ class Import(View):
|
||||||
|
|
||||||
importer = None
|
importer = None
|
||||||
if source == "LibraryThing":
|
if source == "LibraryThing":
|
||||||
importer = librarything_import.LibrarythingImporter()
|
importer = LibrarythingImporter()
|
||||||
else:
|
else:
|
||||||
# Default : GoodReads
|
# Default : GoodReads
|
||||||
importer = goodreads_import.GoodreadsImporter()
|
importer = GoodreadsImporter()
|
||||||
|
|
||||||
try:
|
try:
|
||||||
job = importer.create_job(
|
job = importer.create_job(
|
||||||
|
@ -89,10 +90,11 @@ class ImportStatus(View):
|
||||||
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))
|
items.append(get_object_or_404(models.ImportItem, id=item))
|
||||||
|
|
||||||
job = goodreads_import.create_retry_job(
|
importer = Importer()
|
||||||
|
job = importer.create_retry_job(
|
||||||
request.user,
|
request.user,
|
||||||
job,
|
job,
|
||||||
items,
|
items,
|
||||||
)
|
)
|
||||||
goodreads_import.start_import(job)
|
importer.start_import(job)
|
||||||
return redirect("/import/%d" % job.id)
|
return redirect("/import/%d" % job.id)
|
||||||
|
|
|
@ -27,17 +27,17 @@ class Lists(View):
|
||||||
except ValueError:
|
except ValueError:
|
||||||
page = 1
|
page = 1
|
||||||
|
|
||||||
user = request.user if request.user.is_authenticated else None
|
|
||||||
# hide lists with no approved books
|
# hide lists with no approved books
|
||||||
lists = (
|
lists = (
|
||||||
models.List.objects.filter(
|
models.List.objects.annotate(
|
||||||
~Q(user=user),
|
item_count=Count("listitem", filter=Q(listitem__approved=True))
|
||||||
)
|
)
|
||||||
.annotate(item_count=Count("listitem", filter=Q(listitem__approved=True)))
|
|
||||||
.filter(item_count__gt=0)
|
.filter(item_count__gt=0)
|
||||||
|
.order_by("-updated_date")
|
||||||
.distinct()
|
.distinct()
|
||||||
.all()
|
.all()
|
||||||
)
|
)
|
||||||
|
|
||||||
lists = privacy_filter(
|
lists = privacy_filter(
|
||||||
request.user, lists, privacy_levels=["public", "followers"]
|
request.user, lists, privacy_levels=["public", "followers"]
|
||||||
)
|
)
|
||||||
|
|
Loading…
Reference in a new issue