Change how goodread import writes reviews

- adds published date
 - broadcasts review imports
 - completes review and shelve actions as it goes
 - some small connector fixes

 fixes #247
This commit is contained in:
Mouse Reeve 2020-10-29 14:29:31 -07:00
parent 7febcec229
commit a46d7f5dc7
5 changed files with 67 additions and 57 deletions

View file

@ -2,14 +2,16 @@
from abc import ABC, abstractmethod from abc import ABC, abstractmethod
from dateutil import parser from dateutil import parser
import pytz import pytz
from urllib3.exceptions import ProtocolError
import requests import requests
from requests import HTTPError
from django.db import transaction from django.db import transaction
from bookwyrm import models from bookwyrm import models
class ConnectorException(Exception): class ConnectorException(HTTPError):
''' when the connector can't do what was asked ''' ''' when the connector can't do what was asked '''
@ -155,9 +157,11 @@ class AbstractConnector(ABC):
''' for creating a new book or syncing with data ''' ''' for creating a new book or syncing with data '''
book = update_from_mappings(book, data, self.book_mappings) book = update_from_mappings(book, data, self.book_mappings)
author_text = []
for author in self.get_authors_from_data(data): for author in self.get_authors_from_data(data):
book.authors.add(author) book.authors.add(author)
book.author_text = ', '.join(a.display_name for a in book.authors.all()) author_text += author.display_name
book.author_text = ', '.join(author_text)
book.save() book.save()
if not update_cover: if not update_cover:
@ -287,12 +291,15 @@ def get_date(date_string):
def get_data(url): def get_data(url):
''' wrapper for request.get ''' ''' wrapper for request.get '''
try:
resp = requests.get( resp = requests.get(
url, url,
headers={ headers={
'Accept': 'application/json; charset=utf-8', 'Accept': 'application/json; charset=utf-8',
}, },
) )
except ProtocolError:
raise ConnectorException()
if not resp.ok: if not resp.ok:
resp.raise_for_status() resp.raise_for_status()
data = resp.json() data = resp.json()

View file

@ -42,13 +42,10 @@ def import_data(job_id):
if item.book: if item.book:
item.save() item.save()
results.append(item) results.append(item)
# shelves book and handles reviews
outgoing.handle_imported_book(job.user, item)
else: else:
item.fail_reason = "Could not match book on OpenLibrary" item.fail_reason = "Could not find a match for book"
item.save() item.save()
status = outgoing.handle_import_books(job.user, results)
if status:
job.import_status = status
job.save()
finally: finally:
create_notification(job.user, 'IMPORT', related_import=job) create_notification(job.user, 'IMPORT', related_import=job)

View file

@ -40,8 +40,7 @@ class ImportJob(models.Model):
user = models.ForeignKey(User, on_delete=models.CASCADE) user = models.ForeignKey(User, on_delete=models.CASCADE)
created_date = models.DateTimeField(default=timezone.now) created_date = models.DateTimeField(default=timezone.now)
task_id = models.CharField(max_length=100, null=True) task_id = models.CharField(max_length=100, null=True)
import_status = models.ForeignKey(
'Status', null=True, on_delete=models.PROTECT)
class ImportItem(models.Model): class ImportItem(models.Model):
''' a single line of a csv being imported ''' ''' a single line of a csv being imported '''
@ -71,6 +70,8 @@ class ImportItem(models.Model):
return books_manager.get_or_create_book(search_result.key) return books_manager.get_or_create_book(search_result.key)
except ConnectorException: except ConnectorException:
pass pass
return None
def get_book_from_title_author(self): def get_book_from_title_author(self):
''' search by title and author ''' ''' search by title and author '''
@ -84,6 +85,8 @@ class ImportItem(models.Model):
return books_manager.get_or_create_book(search_result.key) return books_manager.get_or_create_book(search_result.key)
except ConnectorException: except ConnectorException:
pass pass
return None
@property @property
def isbn(self): def isbn(self):
@ -95,6 +98,7 @@ class ImportItem(models.Model):
''' the goodreads shelf field ''' ''' the goodreads shelf field '''
if self.data['Exclusive Shelf']: if self.data['Exclusive Shelf']:
return GOODREADS_SHELVES.get(self.data['Exclusive Shelf']) return GOODREADS_SHELVES.get(self.data['Exclusive Shelf'])
return None
@property @property
def review(self): def review(self):
@ -111,12 +115,14 @@ class ImportItem(models.Model):
''' when the book was added to this dataset ''' ''' when the book was added to this dataset '''
if self.data['Date Added']: if self.data['Date Added']:
return dateutil.parser.parse(self.data['Date Added']) return dateutil.parser.parse(self.data['Date Added'])
return None
@property @property
def date_read(self): def date_read(self):
''' the date a book was completed ''' ''' the date a book was completed '''
if self.data['Date Read']: if self.data['Date Read']:
return dateutil.parser.parse(self.data['Date Read']) return dateutil.parser.parse(self.data['Date Read'])
return None
@property @property
def reads(self): def reads(self):
@ -126,6 +132,7 @@ class ImportItem(models.Model):
return [ReadThrough(start_date=self.date_added)] return [ReadThrough(start_date=self.date_added)]
if self.date_read: if self.date_read:
return [ReadThrough( return [ReadThrough(
start_date=self.date_added,
finish_date=self.date_read, finish_date=self.date_read,
)] )]
return [] return []

View file

@ -155,51 +155,49 @@ def handle_unshelve(user, book, shelf):
broadcast(user, activity) broadcast(user, activity)
def handle_import_books(user, items): def handle_imported_book(user, item):
''' process a goodreads csv and then post about it ''' ''' process a goodreads csv and then post about it '''
new_books = [] if isinstance(item.book, models.Work):
for item in items: item.book = item.book.default_edition
if not item.book:
return
if item.shelf: if item.shelf:
desired_shelf = models.Shelf.objects.get( desired_shelf = models.Shelf.objects.get(
identifier=item.shelf, identifier=item.shelf,
user=user user=user
) )
if isinstance(item.book, models.Work): # shelve the book if it hasn't been shelved already
item.book = item.book.default_edition
if not item.book:
continue
shelf_book, created = models.ShelfBook.objects.get_or_create( shelf_book, created = models.ShelfBook.objects.get_or_create(
book=item.book, shelf=desired_shelf, added_by=user) book=item.book, shelf=desired_shelf, added_by=user)
if created: if created:
new_books.append(item.book) broadcast(user, shelf_book.to_add_activity(user))
activity = shelf_book.to_add_activity(user)
broadcast(user, activity) # only add new read-throughs if the item isn't already shelved
for read in item.reads:
read.book = item.book
read.user = user
read.save()
if item.rating or item.review: if item.rating or item.review:
review_title = 'Review of {!r} on Goodreads'.format( review_title = 'Review of {!r} on Goodreads'.format(
item.book.title, item.book.title,
) if item.review else '' ) if item.review else ''
models.Review.objects.create( # we don't know the publication date of the review,
# but "now" is a bad guess
published_date_guess = item.date_read or item.date_added
review = models.Review.objects.create(
user=user, user=user,
book=item.book, book=item.book,
name=review_title, name=review_title,
content=item.review, content=item.review,
rating=item.rating, rating=item.rating,
published_date=published_date_guess,
) )
for read in item.reads: # we don't need to send out pure activities because non-bookwyrm
read.book = item.book # instances don't need this data
read.user = user broadcast(user, review.to_create_activity(user))
read.save()
if new_books:
message = 'imported {} books'.format(len(new_books))
status = create_generated_note(user, message, mention_books=new_books)
status.save()
broadcast(user, status.to_create_activity(user))
return status
return None
def handle_delete_status(user, status): def handle_delete_status(user, status):

View file

@ -489,7 +489,8 @@ def book_page(request, book_id):
).values_list('identifier', flat=True) ).values_list('identifier', flat=True)
readthroughs = models.ReadThrough.objects.filter( readthroughs = models.ReadThrough.objects.filter(
user=request.user user=request.user,
book=book,
).order_by('start_date') ).order_by('start_date')