moviewyrm/bookwyrm/view_actions.py

881 lines
26 KiB
Python
Raw Normal View History

2020-02-22 22:02:03 +00:00
''' views for actions you can take in the application '''
2020-04-02 17:23:34 +00:00
from io import BytesIO, TextIOWrapper
from uuid import uuid4
2020-04-02 17:23:34 +00:00
from PIL import Image
2020-04-01 14:18:45 +00:00
2020-10-30 17:40:05 +00:00
import dateutil.parser
from dateutil.parser import ParserError
2020-03-15 21:15:36 +00:00
from django.contrib.auth import authenticate, login, logout
2020-10-01 19:59:38 +00:00
from django.contrib.auth.decorators import login_required, permission_required
from django.core.exceptions import PermissionDenied
2020-04-02 17:23:34 +00:00
from django.core.files.base import ContentFile
2020-12-12 00:57:38 +00:00
from django.db import transaction
2020-03-28 22:06:16 +00:00
from django.http import HttpResponseBadRequest, HttpResponseNotFound
2020-11-11 04:11:21 +00:00
from django.shortcuts import get_object_or_404, redirect
2020-02-22 22:02:03 +00:00
from django.template.response import TemplateResponse
from django.utils import timezone
2020-11-28 16:22:25 +00:00
from django.views.decorators.http import require_GET, require_POST
2020-02-22 22:02:03 +00:00
2021-01-02 16:14:28 +00:00
from bookwyrm import forms, models, outgoing, goodreads_import
from bookwyrm.connectors import connector_manager
from bookwyrm.broadcast import broadcast
2020-10-02 20:32:19 +00:00
from bookwyrm.emailing import password_reset_email
from bookwyrm.settings import DOMAIN
2021-01-02 16:14:28 +00:00
from bookwyrm.views import get_user_from_username, get_edition
2020-02-22 22:02:03 +00:00
2020-11-28 17:17:24 +00:00
@require_POST
2020-03-15 21:15:36 +00:00
def user_login(request):
''' authenticate user login '''
login_form = forms.LoginForm(request.POST)
2021-01-04 18:48:48 +00:00
localname = login_form.data['localname']
username = '%s@%s' % (localname, DOMAIN)
password = login_form.data['password']
2020-03-15 21:15:36 +00:00
user = authenticate(request, username=username, password=password)
if user is not None:
# successful login
2020-03-15 21:15:36 +00:00
login(request, user)
user.last_active_date = timezone.now()
2020-03-15 21:15:36 +00:00
return redirect(request.GET.get('next', '/'))
2020-10-04 22:10:34 +00:00
login_form.non_field_errors = 'Username or password are incorrect'
2020-10-26 22:09:51 +00:00
register_form = forms.RegisterForm()
2020-10-04 22:10:34 +00:00
data = {
'login_form': login_form,
'register_form': register_form
}
return TemplateResponse(request, 'login.html', data)
2020-03-15 21:15:36 +00:00
2020-11-28 17:17:24 +00:00
@require_POST
2020-03-15 21:15:36 +00:00
def register(request):
''' join the server '''
if not models.SiteSettings.get().allow_registration:
invite_code = request.POST.get('invite_code')
if not invite_code:
raise PermissionDenied
2020-11-11 05:13:13 +00:00
invite = get_object_or_404(models.SiteInvite, code=invite_code)
2021-01-04 19:14:10 +00:00
if not invite.valid():
raise PermissionDenied
else:
invite = None
2020-03-15 21:15:36 +00:00
form = forms.RegisterForm(request.POST)
2020-10-04 22:10:34 +00:00
errors = False
2020-03-15 21:15:36 +00:00
if not form.is_valid():
2020-10-04 22:10:34 +00:00
errors = True
2020-03-15 21:15:36 +00:00
localname = form.data['localname'].strip()
2020-03-15 21:15:36 +00:00
email = form.data['email']
password = form.data['password']
# check localname and email uniqueness
if models.User.objects.filter(localname=localname).first():
2021-01-04 19:14:10 +00:00
form.errors['localname'] = ['User with this username already exists']
2020-10-04 22:10:34 +00:00
errors = True
if errors:
data = {
'login_form': forms.LoginForm(),
'register_form': form,
'invite': invite,
'valid': invite.valid() if invite else True,
2020-10-04 22:10:34 +00:00
}
if invite:
return TemplateResponse(request, 'invite.html', data)
2020-10-04 22:10:34 +00:00
return TemplateResponse(request, 'login.html', data)
username = '%s@%s' % (localname, DOMAIN)
2020-11-30 18:32:13 +00:00
user = models.User.objects.create_user(
username, email, password, localname=localname, local=True)
if invite:
invite.times_used += 1
invite.save()
2020-03-15 21:15:36 +00:00
login(request, user)
return redirect('/')
@login_required
2020-11-28 16:22:25 +00:00
@require_GET
2020-03-15 21:15:36 +00:00
def user_logout(request):
''' done with this place! outa here! '''
logout(request)
return redirect('/')
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-02 20:32:19 +00:00
def password_reset_request(request):
''' 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')
# remove any existing password reset cods for this user
models.PasswordReset.objects.filter(user=user).all().delete()
# 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)
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-02 20:32:19 +00:00
def password_reset(request):
2020-10-02 21:42:42 +00:00
''' allow a user to change their password through an emailed token '''
2020-10-02 20:32:19 +00:00
try:
reset_code = models.PasswordReset.objects.get(
code=request.POST.get('reset-code')
)
except models.PasswordReset.DoesNotExist:
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')
if new_password != confirm_password:
data = {'errors': ['Passwords do not match']}
return TemplateResponse(request, 'password_reset.html', data)
user.set_password(new_password)
user.save()
login(request, user)
reset_code.delete()
return redirect('/')
2020-10-02 21:42:42 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-02 21:42:42 +00:00
def password_change(request):
''' 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('/user-edit')
request.user.set_password(new_password)
request.user.save()
login(request, request.user)
2020-12-21 22:54:45 +00:00
return redirect('/user/%s' % request.user.localname)
2020-10-02 21:42:42 +00:00
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def edit_profile(request):
''' les get fancy with images '''
2020-12-22 00:19:36 +00:00
form = forms.EditUserForm(
request.POST, request.FILES, instance=request.user)
2020-02-22 22:02:03 +00:00
if not form.is_valid():
2020-12-21 22:54:45 +00:00
data = {'form': form, 'user': request.user}
return TemplateResponse(request, 'edit_user.html', data)
2020-02-22 22:02:03 +00:00
2020-12-21 22:54:45 +00:00
user = form.save(commit=False)
2020-02-22 22:02:03 +00:00
if 'avatar' in form.files:
2020-04-02 17:23:34 +00:00
# crop and resize avatar upload
2020-04-21 01:42:13 +00:00
image = Image.open(form.files['avatar'])
2020-04-02 17:23:34 +00:00
target_size = 120
2020-04-21 01:42:13 +00:00
width, height = image.size
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))
2020-04-02 17:23:34 +00:00
))
output = BytesIO()
2020-04-21 01:42:13 +00:00
cropped.save(output, format=image.format)
2020-04-02 17:23:34 +00:00
ContentFile(output.getvalue())
# set the name to a hash
extension = form.files['avatar'].name.split('.')[-1]
filename = '%s.%s' % (uuid4(), extension)
2020-12-21 22:54:45 +00:00
user.avatar.save(filename, ContentFile(output.getvalue()))
user.save()
2021-01-11 03:51:58 +00:00
broadcast(user, user.to_update_activity(user))
2020-02-22 22:02:03 +00:00
return redirect('/user/%s' % request.user.localname)
@require_POST
2020-05-10 19:56:59 +00:00
def resolve_book(request):
''' figure out the local path to a book from a remote_id '''
remote_id = request.POST.get('remote_id')
2021-01-02 16:14:28 +00:00
connector = connector_manager.get_or_create_connector(remote_id)
2020-11-29 02:46:50 +00:00
book = connector.get_or_create_book(remote_id)
2020-05-10 19:56:59 +00:00
return redirect('/book/%d' % book.id)
2020-03-28 22:06:16 +00:00
@login_required
2020-10-01 19:59:38 +00:00
@permission_required('bookwyrm.edit_book', raise_exception=True)
2020-11-28 16:22:25 +00:00
@require_POST
2020-03-28 22:06:16 +00:00
def edit_book(request, book_id):
''' edit a book cool '''
2020-11-11 05:13:13 +00:00
book = get_object_or_404(models.Edition, id=book_id)
2020-03-28 22:06:16 +00:00
2020-04-02 15:44:53 +00:00
form = forms.EditionForm(request.POST, request.FILES, instance=book)
2020-03-28 22:06:16 +00:00
if not form.is_valid():
data = {
'title': 'Edit Book',
'book': book,
'form': form
}
return TemplateResponse(request, 'edit_book.html', data)
book = form.save()
2020-03-28 22:06:16 +00:00
2021-01-11 03:51:58 +00:00
broadcast(request.user, book.to_update_activity(request.user))
2020-05-04 00:53:14 +00:00
return redirect('/book/%s' % book.id)
2020-03-28 22:06:16 +00:00
2020-12-12 00:57:38 +00:00
@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')
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
2020-12-12 00:57:38 +00:00
)
for shelfbook in shelfbooks.all():
broadcast(request.user, shelfbook.to_remove_activity(request.user))
2020-12-12 00:57:38 +00:00
shelfbook.book = new_edition
shelfbook.save()
broadcast(request.user, shelfbook.to_add_activity(request.user))
2020-12-12 00:57:38 +00:00
readthroughs = models.ReadThrough.objects.filter(
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)
2020-03-28 22:06:16 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-03-28 22:06:16 +00:00
def upload_cover(request, book_id):
''' upload a new cover '''
2020-11-11 05:13:13 +00:00
book = get_object_or_404(models.Edition, id=book_id)
2020-03-28 22:06:16 +00:00
form = forms.CoverForm(request.POST, request.FILES, instance=book)
if not form.is_valid():
2020-11-28 15:55:31 +00:00
return redirect('/book/%d' % book.id)
2020-03-28 22:06:16 +00:00
book.cover = form.files['cover']
book.save()
2021-01-11 03:51:58 +00:00
broadcast(request.user, book.to_update_activity(request.user))
2020-05-04 00:53:14 +00:00
return redirect('/book/%s' % book.id)
2020-03-28 22:06:16 +00:00
2020-11-10 22:52:04 +00:00
@login_required
@require_POST
@permission_required('bookwyrm.edit_book', raise_exception=True)
2020-11-28 15:55:31 +00:00
def add_description(request, book_id):
''' 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')
book.description = description
book.save()
2021-01-11 03:51:58 +00:00
broadcast(request.user, book.to_update_activity(request.user))
2020-11-28 15:55:31 +00:00
return redirect('/book/%s' % book.id)
@login_required
@permission_required('bookwyrm.edit_book', raise_exception=True)
@require_POST
def edit_author(request, author_id):
''' 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)
author = form.save()
2021-01-11 03:51:58 +00:00
broadcast(request.user, author.to_update_activity(request.user))
return redirect('/author/%s' % author.id)
2020-11-10 22:52:04 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-10 22:52:04 +00:00
def create_shelf(request):
''' user generated shelves '''
form = forms.ShelfForm(request.POST)
2020-11-11 04:11:21 +00:00
if not form.is_valid():
return redirect(request.headers.get('Referer', '/'))
shelf = form.save()
return redirect('/user/%s/shelf/%s' % \
(request.user.localname, shelf.identifier))
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-11 04:11:21 +00:00
def edit_shelf(request, shelf_id):
''' user generated shelves '''
shelf = get_object_or_404(models.Shelf, id=shelf_id)
2020-11-11 04:33:46 +00:00
if request.user != shelf.user:
return HttpResponseBadRequest()
2021-01-11 22:32:04 +00:00
if not shelf.editable and request.POST.get('name') != shelf.name:
return HttpResponseBadRequest()
2020-11-11 04:11:21 +00:00
form = forms.ShelfForm(request.POST, instance=shelf)
2020-11-10 22:52:04 +00:00
if not form.is_valid():
2021-01-11 22:23:56 +00:00
return redirect(shelf.local_path)
2020-11-10 22:52:04 +00:00
shelf = form.save()
2021-01-11 22:23:56 +00:00
return redirect(shelf.local_path)
2020-11-10 22:52:04 +00:00
2020-11-11 04:33:46 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-11 04:33:46 +00:00
def delete_shelf(request, shelf_id):
''' 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)
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def shelve(request):
2020-03-28 22:06:16 +00:00
''' put a on a user's shelf '''
2021-01-02 16:14:28 +00:00
book = get_edition(request.POST['book'])
2020-03-30 22:03:21 +00:00
2020-02-22 22:02:03 +00:00
desired_shelf = models.Shelf.objects.filter(
identifier=request.POST['shelf'],
user=request.user
).first()
if request.POST.get('reshelve', True):
try:
current_shelf = models.Shelf.objects.get(
user=request.user,
2020-03-30 22:03:21 +00:00
edition=book
2020-02-22 22:02:03 +00:00
)
outgoing.handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
outgoing.handle_shelve(request.user, book, desired_shelf)
2020-11-06 17:07:04 +00:00
# post about "want to read" shelves
if desired_shelf.identifier == 'to-read':
outgoing.handle_reading_status(
request.user,
desired_shelf,
book,
privacy=desired_shelf.privacy
2020-11-06 17:07:04 +00:00
)
2020-02-22 22:02:03 +00:00
return redirect('/')
2020-10-28 23:52:23 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-28 23:52:23 +00:00
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'])
outgoing.handle_unshelve(request.user, book, current_shelf)
return redirect(request.headers.get('Referer', '/'))
2020-11-06 16:51:50 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-11 05:45:22 +00:00
def start_reading(request, book_id):
2020-11-06 16:51:50 +00:00
''' begin reading a book '''
2021-01-02 16:14:28 +00:00
book = get_edition(book_id)
2020-11-06 16:51:50 +00:00
shelf = models.Shelf.objects.filter(
identifier='reading',
user=request.user
).first()
# create a readthrough
readthrough = update_readthrough(request, book=book)
if readthrough.start_date:
readthrough.save()
# create a progress update if we have a page
readthrough.create_update()
2020-11-06 16:51:50 +00:00
# shelve the book
if request.POST.get('reshelve', True):
try:
current_shelf = models.Shelf.objects.get(
2020-11-06 17:07:04 +00:00
user=request.user,
2020-11-06 16:51:50 +00:00
edition=book
)
outgoing.handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
outgoing.handle_shelve(request.user, book, shelf)
# post about it (if you want)
if request.POST.get('post-status'):
privacy = request.POST.get('privacy')
outgoing.handle_reading_status(request.user, shelf, book, privacy)
return redirect(request.headers.get('Referer', '/'))
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-11 05:45:22 +00:00
def finish_reading(request, book_id):
2020-11-06 16:51:50 +00:00
''' a user completed a book, yay '''
2021-01-02 16:14:28 +00:00
book = get_edition(book_id)
2020-11-06 16:51:50 +00:00
shelf = models.Shelf.objects.filter(
identifier='read',
user=request.user
).first()
# update or create a readthrough
readthrough = update_readthrough(request, book=book)
if readthrough.start_date or readthrough.finish_date:
readthrough.save()
# shelve the book
if request.POST.get('reshelve', True):
try:
current_shelf = models.Shelf.objects.get(
user=request.user,
edition=book
)
outgoing.handle_unshelve(request.user, book, current_shelf)
except models.Shelf.DoesNotExist:
# this just means it isn't currently on the user's shelves
pass
outgoing.handle_shelve(request.user, book, shelf)
# post about it (if you want)
if request.POST.get('post-status'):
privacy = request.POST.get('privacy')
outgoing.handle_reading_status(request.user, shelf, book, privacy)
return redirect(request.headers.get('Referer', '/'))
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-06 16:51:50 +00:00
def edit_readthrough(request):
''' can't use the form because the dates are too finnicky '''
readthrough = update_readthrough(request, create=False)
if not readthrough:
return HttpResponseNotFound()
# don't let people edit other people's data
if request.user != readthrough.user:
return HttpResponseBadRequest()
readthrough.save()
# record the progress update individually
# use default now for date field
readthrough.create_update()
2020-11-06 16:51:50 +00:00
return redirect(request.headers.get('Referer', '/'))
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-06 16:51:50 +00:00
def delete_readthrough(request):
''' remove a readthrough '''
2020-11-11 05:13:13 +00:00
readthrough = get_object_or_404(
models.ReadThrough, id=request.POST.get('id'))
2020-11-06 16:51:50 +00:00
# don't let people edit other people's data
if request.user != readthrough.user:
return HttpResponseBadRequest()
readthrough.delete()
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'))
readthrough = update_readthrough(request, create=True, book=book)
if not readthrough:
return redirect(book.local_path)
readthrough.save()
return redirect(request.headers.get('Referer', '/'))
2020-04-03 19:43:49 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-04-03 19:43:49 +00:00
def rate(request):
''' just a star rating for a book '''
form = forms.RatingForm(request.POST)
2020-10-26 22:00:15 +00:00
return handle_status(request, form)
2020-04-03 19:43:49 +00:00
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def review(request):
2020-03-21 23:50:49 +00:00
''' create a book review '''
2020-02-22 22:02:03 +00:00
form = forms.ReviewForm(request.POST)
2020-10-26 22:00:15 +00:00
return handle_status(request, form)
2020-02-22 22:02:03 +00:00
2020-04-08 16:40:47 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-04-08 16:40:47 +00:00
def quotate(request):
''' create a book quotation '''
form = forms.QuotationForm(request.POST)
2020-10-26 22:00:15 +00:00
return handle_status(request, form)
2020-04-08 16:40:47 +00:00
2020-03-21 23:50:49 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-03-21 23:50:49 +00:00
def comment(request):
''' create a book comment '''
form = forms.CommentForm(request.POST)
2020-10-26 22:00:15 +00:00
return handle_status(request, form)
2020-10-27 18:32:15 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-27 18:32:15 +00:00
def reply(request):
''' respond to a book review '''
form = forms.ReplyForm(request.POST)
return handle_status(request, form)
2020-10-26 22:00:15 +00:00
def handle_status(request, form):
2020-10-27 18:32:15 +00:00
''' all the "create a status" functions are the same '''
2020-03-21 23:50:49 +00:00
if not form.is_valid():
2020-10-27 18:32:15 +00:00
return redirect(request.headers.get('Referer', '/'))
2020-03-21 23:50:49 +00:00
2020-10-26 22:00:15 +00:00
outgoing.handle_status(request.user, form)
2020-10-27 18:32:15 +00:00
return redirect(request.headers.get('Referer', '/'))
2020-03-21 23:50:49 +00:00
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def tag(request):
''' 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')
2020-11-11 05:13:13 +00:00
book = get_object_or_404(models.Edition, id=book_id)
2020-11-06 22:25:48 +00:00
tag_obj, created = models.Tag.objects.get_or_create(
name=name,
)
2020-12-31 16:30:51 +00:00
user_tag, _ = models.UserTag.objects.get_or_create(
user=request.user,
2020-11-06 22:25:48 +00:00
book=book,
tag=tag_obj,
2020-11-06 22:25:48 +00:00
)
2020-02-22 22:02:03 +00:00
2020-11-06 22:25:48 +00:00
if created:
2020-12-31 16:30:51 +00:00
broadcast(request.user, user_tag.to_add_activity(request.user))
return redirect('/book/%s' % book_id)
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def untag(request):
''' untag a book '''
name = request.POST.get('name')
2021-01-01 15:27:56 +00:00
tag_obj = get_object_or_404(models.Tag, name=name)
book_id = request.POST.get('book')
2020-12-31 16:42:09 +00:00
book = get_object_or_404(models.Edition, id=book_id)
2021-01-01 15:27:56 +00:00
user_tag = get_object_or_404(
models.UserTag, tag=tag_obj, book=book, user=request.user)
tag_activity = user_tag.to_remove_activity(request.user)
user_tag.delete()
2020-02-22 22:02:03 +00:00
2020-12-31 16:42:09 +00:00
broadcast(request.user, tag_activity)
return redirect('/book/%s' % book_id)
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def favorite(request, status_id):
''' like a status '''
status = models.Status.objects.get(id=status_id)
outgoing.handle_favorite(request.user, status)
2020-03-21 22:21:27 +00:00
return redirect(request.headers.get('Referer', '/'))
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-03-21 22:21:27 +00:00
def unfavorite(request, status_id):
''' like a status '''
status = models.Status.objects.get(id=status_id)
outgoing.handle_unfavorite(request.user, status)
2020-02-22 22:02:03 +00:00
return redirect(request.headers.get('Referer', '/'))
2020-11-08 02:31:01 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
def boost(request, status_id):
''' boost a status '''
status = models.Status.objects.get(id=status_id)
outgoing.handle_boost(request.user, status)
return redirect(request.headers.get('Referer', '/'))
2020-02-22 22:02:03 +00:00
2020-10-08 19:32:45 +00:00
2020-11-08 02:31:01 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-08 02:31:01 +00:00
def unboost(request, status_id):
''' boost a status '''
status = models.Status.objects.get(id=status_id)
outgoing.handle_unboost(request.user, status)
return redirect(request.headers.get('Referer', '/'))
2020-10-08 19:32:45 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
def delete_status(request, status_id):
2020-10-08 19:32:45 +00:00
''' delete and tombstone a status '''
2020-11-11 05:13:13 +00:00
status = get_object_or_404(models.Status, id=status_id)
2020-10-08 19:32:45 +00:00
# don't let people delete other people's statuses
if status.user != request.user:
return HttpResponseBadRequest()
# perform deletion
outgoing.handle_delete_status(request.user, status)
return redirect(request.headers.get('Referer', '/'))
2020-02-22 22:02:03 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def follow(request):
''' follow another user, here or abroad '''
username = request.POST['user']
try:
to_follow = get_user_from_username(username)
except models.User.DoesNotExist:
return HttpResponseBadRequest()
outgoing.handle_follow(request.user, to_follow)
2020-02-22 22:02:03 +00:00
user_slug = to_follow.localname if to_follow.localname \
else to_follow.username
return redirect('/user/%s' % user_slug)
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-02-22 22:02:03 +00:00
def unfollow(request):
''' unfollow a user '''
username = request.POST['user']
try:
to_unfollow = get_user_from_username(username)
except models.User.DoesNotExist:
return HttpResponseBadRequest()
outgoing.handle_unfollow(request.user, to_unfollow)
2020-02-22 22:02:03 +00:00
user_slug = to_unfollow.localname if to_unfollow.localname \
else to_unfollow.username
return redirect('/user/%s' % user_slug)
2020-03-07 22:50:29 +00:00
@login_required
def clear_notifications(request):
2020-03-15 21:15:36 +00:00
''' permanently delete notification for user '''
2020-03-07 22:50:29 +00:00
request.user.notification_set.filter(read=True).delete()
return redirect('/notifications')
2020-03-15 21:15:36 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
def accept_follow_request(request):
2020-03-15 21:15:36 +00:00
''' a user accepts a follow request '''
username = request.POST['user']
try:
requester = get_user_from_username(username)
except models.User.DoesNotExist:
return HttpResponseBadRequest()
try:
2020-03-15 21:15:36 +00:00
follow_request = models.UserFollowRequest.objects.get(
user_subject=requester,
user_object=request.user
)
except models.UserFollowRequest.DoesNotExist:
# Request already dealt with.
pass
else:
2020-10-16 21:14:07 +00:00
outgoing.handle_accept(follow_request)
return redirect('/user/%s' % request.user.localname)
2020-03-15 21:15:36 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
def delete_follow_request(request):
2020-03-15 21:15:36 +00:00
''' a user rejects a follow request '''
username = request.POST['user']
try:
requester = get_user_from_username(username)
except models.User.DoesNotExist:
return HttpResponseBadRequest()
try:
2020-03-15 21:15:36 +00:00
follow_request = models.UserFollowRequest.objects.get(
user_subject=requester,
user_object=request.user
)
except models.UserFollowRequest.DoesNotExist:
return HttpResponseBadRequest()
2020-10-16 21:28:25 +00:00
outgoing.handle_reject(follow_request)
return redirect('/user/%s' % request.user.localname)
2020-03-27 16:33:31 +00:00
2020-03-23 16:43:11 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-03-23 16:43:11 +00:00
def import_data(request):
2020-03-27 16:33:31 +00:00
''' ingest a goodreads csv '''
2020-03-23 16:43:11 +00:00
form = forms.ImportForm(request.POST, request.FILES)
if form.is_valid():
include_reviews = request.POST.get('include_reviews') == 'on'
privacy = request.POST.get('privacy')
try:
job = goodreads_import.create_job(
request.user,
TextIOWrapper(
request.FILES['csv_file'],
encoding=request.encoding),
include_reviews,
privacy,
)
except (UnicodeDecodeError, ValueError):
return HttpResponseBadRequest('Not a valid csv file')
goodreads_import.start_import(job)
2020-11-13 17:02:41 +00:00
return redirect('/import-status/%d' % job.id)
2020-04-01 14:18:45 +00:00
return HttpResponseBadRequest()
2020-06-03 16:38:30 +00:00
2020-10-01 19:59:38 +00:00
2020-11-13 17:02:41 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-11-13 17:02:41 +00:00
def retry_import(request):
''' ingest a goodreads csv '''
job = get_object_or_404(models.ImportJob, id=request.POST.get('import_job'))
items = []
for item in request.POST.getlist('import_item'):
items.append(get_object_or_404(models.ImportItem, id=item))
job = goodreads_import.create_retry_job(
request.user,
job,
items,
)
goodreads_import.start_import(job)
return redirect('/import-status/%d' % job.id)
2020-06-03 16:38:30 +00:00
@login_required
2020-11-28 16:22:25 +00:00
@require_POST
2020-10-01 19:59:38 +00:00
@permission_required('bookwyrm.create_invites', raise_exception=True)
2020-06-03 16:38:30 +00:00
def create_invite(request):
2020-09-21 17:25:26 +00:00
''' creates a user invite database entry '''
2020-06-03 16:38:30 +00:00
form = forms.CreateInviteForm(request.POST)
if not form.is_valid():
return HttpResponseBadRequest("ERRORS : %s" % (form.errors,))
invite = form.save(commit=False)
invite.user = request.user
invite.save()
2020-09-30 04:45:59 +00:00
return redirect('/invite')
2020-11-06 16:51:50 +00:00
def update_readthrough(request, book=None, create=True):
''' updates but does not save dates on a readthrough '''
try:
read_id = request.POST.get('id')
if not read_id:
raise models.ReadThrough.DoesNotExist
readthrough = models.ReadThrough.objects.get(id=read_id)
except models.ReadThrough.DoesNotExist:
if not create or not book:
return None
readthrough = models.ReadThrough(
user=request.user,
book=book,
)
start_date = request.POST.get('start_date')
if start_date:
try:
2021-01-11 00:11:02 +00:00
start_date = timezone.make_aware(dateutil.parser.parse(start_date))
2020-11-06 16:51:50 +00:00
readthrough.start_date = start_date
except ParserError:
pass
finish_date = request.POST.get('finish_date')
if finish_date:
try:
finish_date = timezone.make_aware(
dateutil.parser.parse(finish_date))
2020-11-06 16:51:50 +00:00
readthrough.finish_date = finish_date
except ParserError:
pass
progress = request.POST.get('progress')
if progress:
try:
progress = int(progress)
readthrough.progress = progress
except ValueError:
pass
progress_mode = request.POST.get('progress_mode')
if progress_mode:
try:
progress_mode = models.ProgressMode(progress_mode)
readthrough.progress_mode = progress_mode
except ValueError:
pass
if not readthrough.start_date and not readthrough.finish_date:
return None
2020-11-06 16:51:50 +00:00
return readthrough