2021-10-01 05:22:32 +00:00
|
|
|
""" the good stuff! the books! """
|
2022-02-26 00:40:34 +00:00
|
|
|
from re import sub, findall
|
2021-10-01 05:22:32 +00:00
|
|
|
from django.contrib.auth.decorators import login_required, permission_required
|
|
|
|
from django.contrib.postgres.search import SearchRank, SearchVector
|
|
|
|
from django.db import transaction
|
|
|
|
from django.http import HttpResponseBadRequest
|
|
|
|
from django.shortcuts import get_object_or_404, redirect
|
|
|
|
from django.template.response import TemplateResponse
|
|
|
|
from django.utils.decorators import method_decorator
|
2022-02-25 19:50:25 +00:00
|
|
|
from django.views.decorators.http import require_POST
|
2021-10-01 05:22:32 +00:00
|
|
|
from django.views import View
|
|
|
|
|
2021-10-03 18:55:16 +00:00
|
|
|
from bookwyrm import book_search, forms, models
|
2021-11-22 01:52:59 +00:00
|
|
|
|
2021-11-21 08:55:55 +00:00
|
|
|
# from bookwyrm.activitypub.base_activity import ActivityObject
|
2021-11-01 08:50:49 +00:00
|
|
|
from bookwyrm.utils.isni import (
|
|
|
|
find_authors_by_name,
|
2021-11-21 08:55:55 +00:00
|
|
|
build_author_from_isni,
|
2021-11-01 08:50:49 +00:00
|
|
|
augment_author_metadata,
|
|
|
|
)
|
2021-10-01 05:22:32 +00:00
|
|
|
from bookwyrm.views.helpers import get_edition
|
|
|
|
from .books import set_cover_from_url
|
|
|
|
|
|
|
|
# pylint: disable=no-self-use
|
|
|
|
@method_decorator(login_required, name="dispatch")
|
|
|
|
@method_decorator(
|
|
|
|
permission_required("bookwyrm.edit_book", raise_exception=True), name="dispatch"
|
|
|
|
)
|
|
|
|
class EditBook(View):
|
|
|
|
"""edit a book"""
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
def get(self, request, book_id):
|
2021-10-01 05:22:32 +00:00
|
|
|
"""info about a book"""
|
2022-02-26 00:40:34 +00:00
|
|
|
book = get_edition(book_id)
|
|
|
|
if not book.description:
|
|
|
|
book.description = book.parent_work.description
|
2021-10-01 05:22:32 +00:00
|
|
|
data = {"book": book, "form": forms.EditionForm(instance=book)}
|
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
def post(self, request, book_id):
|
2021-10-01 05:22:32 +00:00
|
|
|
"""edit a book cool"""
|
2022-02-26 01:55:30 +00:00
|
|
|
book = get_object_or_404(models.Edition, id=book_id)
|
2021-10-01 05:22:32 +00:00
|
|
|
form = forms.EditionForm(request.POST, request.FILES, instance=book)
|
|
|
|
|
|
|
|
data = {"book": book, "form": form}
|
|
|
|
if not form.is_valid():
|
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
data = add_authors(request, data)
|
2021-10-01 05:22:32 +00:00
|
|
|
|
|
|
|
# either of the above cases requires additional confirmation
|
2022-02-26 00:40:34 +00:00
|
|
|
if data.get("add_author"):
|
2021-10-01 05:22:32 +00:00
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
|
|
|
remove_authors = request.POST.getlist("remove_authors")
|
|
|
|
for author_id in remove_authors:
|
|
|
|
book.authors.remove(author_id)
|
|
|
|
|
|
|
|
book = form.save(commit=False)
|
2022-02-26 00:40:34 +00:00
|
|
|
|
2021-10-01 05:22:32 +00:00
|
|
|
url = request.POST.get("cover-url")
|
|
|
|
if url:
|
|
|
|
image = set_cover_from_url(url)
|
|
|
|
if image:
|
|
|
|
book.cover.save(*image, save=False)
|
2022-02-26 00:40:34 +00:00
|
|
|
|
2021-10-01 05:22:32 +00:00
|
|
|
book.save()
|
|
|
|
return redirect(f"/book/{book.id}")
|
|
|
|
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
@method_decorator(login_required, name="dispatch")
|
|
|
|
@method_decorator(
|
|
|
|
permission_required("bookwyrm.edit_book", raise_exception=True), name="dispatch"
|
|
|
|
)
|
|
|
|
class CreateBook(View):
|
|
|
|
"""brand new book"""
|
|
|
|
|
|
|
|
def get(self, request):
|
|
|
|
"""info about a book"""
|
|
|
|
data = {"form": forms.EditionForm()}
|
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
|
|
|
# pylint: disable=too-many-locals
|
|
|
|
def post(self, request):
|
|
|
|
"""create a new book"""
|
|
|
|
# returns None if no match is found
|
|
|
|
form = forms.EditionForm(request.POST, request.FILES)
|
|
|
|
data = {"form": form}
|
2022-02-26 01:55:30 +00:00
|
|
|
|
|
|
|
# collect data provided by the work or import item
|
|
|
|
parent_work_id = request.POST.get("parent_work")
|
|
|
|
authors = None
|
|
|
|
if request.POST.get("authors"):
|
|
|
|
author_ids = findall(r"\d+", request.POST["authors"])
|
|
|
|
authors = models.Author.objects.filter(id__in=author_ids)
|
|
|
|
|
|
|
|
# fake book in case we need to keep editing
|
|
|
|
if parent_work_id:
|
|
|
|
data["book"] = {
|
|
|
|
"parent_work": {"id": parent_work_id},
|
|
|
|
"authors": authors,
|
|
|
|
}
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
if not form.is_valid():
|
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
|
|
|
data = add_authors(request, data)
|
|
|
|
|
|
|
|
# check if this is an edition of an existing work
|
2022-02-26 01:55:30 +00:00
|
|
|
author_text = ", ".join(data.get("add_author", []))
|
2022-02-26 00:40:34 +00:00
|
|
|
data["book_matches"] = book_search.search(
|
|
|
|
f'{form.cleaned_data.get("title")} {author_text}',
|
|
|
|
min_confidence=0.1,
|
|
|
|
)[:5]
|
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
# go to confirm mode
|
|
|
|
if not parent_work_id or data.get("add_author"):
|
2022-05-23 20:02:06 +00:00
|
|
|
data["confirm_mode"] = True
|
2022-02-26 00:40:34 +00:00
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
with transaction.atomic():
|
|
|
|
book = form.save()
|
|
|
|
parent_work = get_object_or_404(models.Work, id=parent_work_id)
|
|
|
|
book.parent_work = parent_work
|
2022-02-26 00:40:34 +00:00
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
if authors:
|
|
|
|
book.authors.add(*authors)
|
2022-02-26 00:40:34 +00:00
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
url = request.POST.get("cover-url")
|
|
|
|
if url:
|
|
|
|
image = set_cover_from_url(url)
|
|
|
|
if image:
|
|
|
|
book.cover.save(*image, save=False)
|
|
|
|
|
|
|
|
book.save()
|
2022-02-26 00:40:34 +00:00
|
|
|
return redirect(f"/book/{book.id}")
|
|
|
|
|
2022-02-26 01:23:13 +00:00
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
def add_authors(request, data):
|
|
|
|
"""helper for adding authors"""
|
|
|
|
add_author = [author for author in request.POST.getlist("add_author") if author]
|
|
|
|
if not add_author:
|
|
|
|
return data
|
|
|
|
|
|
|
|
data["add_author"] = add_author
|
|
|
|
data["author_matches"] = []
|
|
|
|
data["isni_matches"] = []
|
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
# creting a book or adding an author to a book needs another step
|
|
|
|
data["confirm_mode"] = True
|
|
|
|
# this isn't preserved because it isn't part of the form obj
|
|
|
|
data["remove_authors"] = request.POST.getlist("remove_authors")
|
|
|
|
data["cover_url"] = request.POST.get("cover-url")
|
|
|
|
|
2022-02-26 00:40:34 +00:00
|
|
|
for author in add_author:
|
|
|
|
# filter out empty author fields
|
|
|
|
if not author:
|
|
|
|
continue
|
|
|
|
# check for existing authors
|
2022-02-26 01:23:13 +00:00
|
|
|
vector = SearchVector("name", weight="A") + SearchVector("aliases", weight="B")
|
2022-02-26 00:40:34 +00:00
|
|
|
|
|
|
|
author_matches = (
|
|
|
|
models.Author.objects.annotate(search=vector)
|
|
|
|
.annotate(rank=SearchRank(vector, author))
|
|
|
|
.filter(rank__gt=0.4)
|
|
|
|
.order_by("-rank")[:5]
|
|
|
|
)
|
|
|
|
|
|
|
|
isni_authors = find_authors_by_name(
|
|
|
|
author, description=True
|
|
|
|
) # find matches from ISNI API
|
|
|
|
|
|
|
|
# dedupe isni authors we already have in the DB
|
|
|
|
exists = [
|
|
|
|
i
|
|
|
|
for i in isni_authors
|
|
|
|
for a in author_matches
|
|
|
|
if sub(r"\D", "", str(i.isni)) == sub(r"\D", "", str(a.isni))
|
|
|
|
]
|
|
|
|
|
|
|
|
# pylint: disable=cell-var-from-loop
|
|
|
|
matches = list(filter(lambda x: x not in exists, isni_authors))
|
|
|
|
# combine existing and isni authors
|
|
|
|
matches.extend(author_matches)
|
|
|
|
|
|
|
|
data["author_matches"].append(
|
|
|
|
{
|
|
|
|
"name": author.strip(),
|
|
|
|
"matches": matches,
|
|
|
|
"existing_isnis": exists,
|
|
|
|
}
|
|
|
|
)
|
2022-05-23 20:52:10 +00:00
|
|
|
return data
|
2022-02-26 00:40:34 +00:00
|
|
|
|
|
|
|
|
2022-02-25 19:50:25 +00:00
|
|
|
@require_POST
|
|
|
|
@permission_required("bookwyrm.edit_book", raise_exception=True)
|
|
|
|
def create_book_from_data(request):
|
|
|
|
"""create a book with starter data"""
|
2022-02-26 00:40:34 +00:00
|
|
|
author_ids = findall(r"\d+", request.POST.get("authors"))
|
|
|
|
book = {
|
|
|
|
"parent_work": {"id": request.POST.get("parent_work")},
|
2022-02-26 01:23:13 +00:00
|
|
|
"authors": models.Author.objects.filter(id__in=author_ids).all(),
|
2022-03-17 15:02:59 +00:00
|
|
|
"subjects": request.POST.getlist("subjects"),
|
2022-02-26 00:40:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
data = {"book": book, "form": forms.EditionForm(request.POST)}
|
2022-02-25 19:50:25 +00:00
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
|
|
|
|
2021-10-01 05:22:32 +00:00
|
|
|
@method_decorator(login_required, name="dispatch")
|
|
|
|
@method_decorator(
|
|
|
|
permission_required("bookwyrm.edit_book", raise_exception=True), name="dispatch"
|
|
|
|
)
|
|
|
|
class ConfirmEditBook(View):
|
|
|
|
"""confirm edits to a book"""
|
|
|
|
|
2021-11-01 09:04:25 +00:00
|
|
|
# pylint: disable=too-many-locals
|
2021-11-22 02:01:58 +00:00
|
|
|
# pylint: disable=too-many-branches
|
2021-10-01 05:22:32 +00:00
|
|
|
def post(self, request, book_id=None):
|
|
|
|
"""edit a book cool"""
|
|
|
|
# returns None if no match is found
|
|
|
|
book = models.Edition.objects.filter(id=book_id).first()
|
|
|
|
form = forms.EditionForm(request.POST, request.FILES, instance=book)
|
|
|
|
|
|
|
|
data = {"book": book, "form": form}
|
|
|
|
if not form.is_valid():
|
|
|
|
return TemplateResponse(request, "book/edit/edit_book.html", data)
|
|
|
|
|
|
|
|
with transaction.atomic():
|
|
|
|
# save book
|
|
|
|
book = form.save()
|
|
|
|
|
2022-02-26 01:55:30 +00:00
|
|
|
# add known authors
|
|
|
|
authors = None
|
|
|
|
if request.POST.get("authors"):
|
|
|
|
author_ids = findall(r"\d+", request.POST["authors"])
|
|
|
|
authors = models.Author.objects.filter(id__in=author_ids)
|
|
|
|
book.authors.add(*authors)
|
|
|
|
|
2021-10-01 05:22:32 +00:00
|
|
|
# get or create author as needed
|
|
|
|
for i in range(int(request.POST.get("author-match-count", 0))):
|
|
|
|
match = request.POST.get(f"author_match-{i}")
|
|
|
|
if not match:
|
|
|
|
return HttpResponseBadRequest()
|
|
|
|
try:
|
|
|
|
# if it's an int, it's an ID
|
|
|
|
match = int(match)
|
|
|
|
author = get_object_or_404(
|
|
|
|
models.Author, id=request.POST[f"author_match-{i}"]
|
|
|
|
)
|
2021-11-01 08:50:49 +00:00
|
|
|
# update author metadata if the ISNI record is more complete
|
|
|
|
isni = request.POST.get(f"isni-for-{match}", None)
|
|
|
|
if isni is not None:
|
|
|
|
augment_author_metadata(author, isni)
|
2021-10-01 05:22:32 +00:00
|
|
|
except ValueError:
|
2021-11-21 08:55:55 +00:00
|
|
|
# otherwise it's a new author
|
|
|
|
isni_match = request.POST.get(f"author_match-{i}")
|
|
|
|
author_object = build_author_from_isni(isni_match)
|
2021-11-21 21:49:22 +00:00
|
|
|
# with author data class from isni id
|
2021-11-21 08:55:55 +00:00
|
|
|
if "author" in author_object:
|
2021-11-22 01:52:59 +00:00
|
|
|
skeleton = models.Author.objects.create(
|
|
|
|
name=author_object["author"].name
|
|
|
|
)
|
2021-11-21 08:55:55 +00:00
|
|
|
author = author_object["author"].to_model(
|
2021-11-22 01:52:59 +00:00
|
|
|
model=models.Author, overwrite=True, instance=skeleton
|
2021-11-21 08:55:55 +00:00
|
|
|
)
|
|
|
|
else:
|
2021-11-21 21:49:22 +00:00
|
|
|
# or it's just a name
|
2021-11-21 08:55:55 +00:00
|
|
|
author = models.Author.objects.create(name=match)
|
2021-10-01 05:22:32 +00:00
|
|
|
book.authors.add(author)
|
|
|
|
|
|
|
|
# create work, if needed
|
2022-02-26 00:40:34 +00:00
|
|
|
if not book.parent_work:
|
2021-10-01 05:22:32 +00:00
|
|
|
work_match = request.POST.get("parent_work")
|
|
|
|
if work_match and work_match != "0":
|
|
|
|
work = get_object_or_404(models.Work, id=work_match)
|
|
|
|
else:
|
|
|
|
work = models.Work.objects.create(title=form.cleaned_data["title"])
|
|
|
|
work.authors.set(book.authors.all())
|
|
|
|
book.parent_work = work
|
|
|
|
|
|
|
|
for author_id in request.POST.getlist("remove_authors"):
|
|
|
|
book.authors.remove(author_id)
|
|
|
|
|
|
|
|
# import cover, if requested
|
|
|
|
url = request.POST.get("cover-url")
|
|
|
|
if url:
|
|
|
|
image = set_cover_from_url(url)
|
|
|
|
if image:
|
|
|
|
book.cover.save(*image, save=False)
|
|
|
|
|
|
|
|
# we don't tell the world when creating a book
|
|
|
|
book.save(broadcast=False)
|
|
|
|
|
|
|
|
return redirect(f"/book/{book.id}")
|