2020-02-22 22:02:03 +00:00
|
|
|
''' views for actions you can take in the application '''
|
2020-04-01 14:18:45 +00:00
|
|
|
from io import TextIOWrapper
|
|
|
|
import re
|
|
|
|
|
2020-03-15 21:15:36 +00:00
|
|
|
from django.contrib.auth import authenticate, login, logout
|
2020-02-22 22:02:03 +00:00
|
|
|
from django.contrib.auth.decorators import login_required
|
2020-03-28 22:06:16 +00:00
|
|
|
from django.http import HttpResponseBadRequest, HttpResponseNotFound
|
2020-02-22 22:02:03 +00:00
|
|
|
from django.shortcuts import redirect
|
|
|
|
from django.template.response import TemplateResponse
|
|
|
|
|
2020-03-07 20:22:28 +00:00
|
|
|
from fedireads import forms, models, books_manager, outgoing
|
2020-03-29 02:12:17 +00:00
|
|
|
from fedireads.goodreads_import import GoodreadsCsv
|
2020-03-15 21:15:36 +00:00
|
|
|
from fedireads.settings import DOMAIN
|
2020-02-22 22:02:03 +00:00
|
|
|
from fedireads.views import get_user_from_username
|
2020-04-01 14:10:59 +00:00
|
|
|
from fedireads.books_manager import get_or_create_book
|
2020-02-22 22:02:03 +00:00
|
|
|
|
|
|
|
|
2020-03-15 21:15:36 +00:00
|
|
|
def user_login(request):
|
|
|
|
''' authenticate user login '''
|
|
|
|
if request.method == 'GET':
|
|
|
|
return redirect('/login')
|
|
|
|
|
2020-03-22 18:20:36 +00:00
|
|
|
register_form = forms.RegisterForm()
|
|
|
|
login_form = forms.LoginForm(request.POST)
|
|
|
|
if not login_form.is_valid():
|
|
|
|
return TemplateResponse(
|
|
|
|
request,
|
|
|
|
'login.html',
|
|
|
|
{'login_form': login_form, 'register_form': register_form}
|
|
|
|
)
|
|
|
|
|
|
|
|
username = login_form.data['username']
|
2020-03-15 21:15:36 +00:00
|
|
|
username = '%s@%s' % (username, DOMAIN)
|
2020-03-22 18:20:36 +00:00
|
|
|
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:
|
|
|
|
login(request, user)
|
|
|
|
return redirect(request.GET.get('next', '/'))
|
2020-03-22 18:20:36 +00:00
|
|
|
return TemplateResponse(
|
|
|
|
request,
|
|
|
|
'login.html',
|
|
|
|
{'login_form': login_form, 'register_form': register_form}
|
|
|
|
)
|
2020-03-15 21:15:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
def register(request):
|
|
|
|
''' join the server '''
|
|
|
|
if request.method == 'GET':
|
|
|
|
return redirect('/login')
|
|
|
|
|
|
|
|
form = forms.RegisterForm(request.POST)
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect('/register/')
|
|
|
|
|
|
|
|
username = form.data['username']
|
|
|
|
email = form.data['email']
|
|
|
|
password = form.data['password']
|
|
|
|
|
|
|
|
user = models.User.objects.create_user(username, email, password)
|
|
|
|
login(request, user)
|
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def user_logout(request):
|
|
|
|
''' done with this place! outa here! '''
|
|
|
|
logout(request)
|
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
|
2020-02-22 22:02:03 +00:00
|
|
|
@login_required
|
|
|
|
def edit_profile(request):
|
|
|
|
''' les get fancy with images '''
|
|
|
|
if not request.method == 'POST':
|
|
|
|
return redirect('/user/%s' % request.user.localname)
|
|
|
|
|
|
|
|
form = forms.EditUserForm(request.POST, request.FILES)
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
request.user.name = form.data['name']
|
|
|
|
if 'avatar' in form.files:
|
|
|
|
request.user.avatar = form.files['avatar']
|
|
|
|
request.user.summary = form.data['summary']
|
2020-03-15 21:15:36 +00:00
|
|
|
request.user.manually_approves_followers = \
|
|
|
|
form.cleaned_data['manually_approves_followers']
|
2020-02-22 22:02:03 +00:00
|
|
|
request.user.save()
|
2020-03-29 02:19:13 +00:00
|
|
|
|
|
|
|
outgoing.handle_update_user(request.user)
|
2020-02-22 22:02:03 +00:00
|
|
|
return redirect('/user/%s' % request.user.localname)
|
|
|
|
|
|
|
|
|
2020-03-28 22:06:16 +00:00
|
|
|
@login_required
|
|
|
|
def edit_book(request, book_id):
|
|
|
|
''' edit a book cool '''
|
|
|
|
if not request.method == 'POST':
|
|
|
|
return redirect('/book/%s' % request.user.localname)
|
|
|
|
|
|
|
|
try:
|
|
|
|
book = models.Book.objects.get(id=book_id)
|
|
|
|
except models.Book.DoesNotExist:
|
|
|
|
return HttpResponseNotFound()
|
|
|
|
|
|
|
|
form = forms.BookForm(request.POST, request.FILES, instance=book)
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect(request.headers.get('Referer', '/'))
|
|
|
|
form.save()
|
|
|
|
|
2020-03-29 02:12:17 +00:00
|
|
|
outgoing.handle_update_book(request.user, book)
|
2020-03-28 22:06:16 +00:00
|
|
|
return redirect('/book/%s' % book.fedireads_key)
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def upload_cover(request, book_id):
|
|
|
|
''' upload a new cover '''
|
|
|
|
# TODO: alternate covers?
|
|
|
|
if not request.method == 'POST':
|
|
|
|
return redirect('/book/%s' % request.user.localname)
|
|
|
|
|
|
|
|
try:
|
|
|
|
book = models.Book.objects.get(id=book_id)
|
|
|
|
except models.Book.DoesNotExist:
|
|
|
|
return HttpResponseNotFound()
|
|
|
|
|
|
|
|
form = forms.CoverForm(request.POST, request.FILES, instance=book)
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect(request.headers.get('Referer', '/'))
|
|
|
|
|
|
|
|
book.cover = form.files['cover']
|
|
|
|
book.sync_cover = False
|
|
|
|
book.save()
|
|
|
|
|
2020-03-29 02:12:17 +00:00
|
|
|
outgoing.handle_update_book(request.user, book)
|
2020-03-28 22:06:16 +00:00
|
|
|
return redirect('/book/%s' % book.fedireads_key)
|
|
|
|
|
|
|
|
|
2020-02-22 22:02:03 +00:00
|
|
|
@login_required
|
|
|
|
def shelve(request):
|
2020-03-28 22:06:16 +00:00
|
|
|
''' put a on a user's shelf '''
|
2020-03-30 22:03:21 +00:00
|
|
|
book = models.Book.objects.select_subclasses().get(id=request.POST['book'])
|
|
|
|
if isinstance(book, models.Work):
|
|
|
|
book = book.default_edition
|
|
|
|
|
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(
|
2020-03-30 22:27:54 +00:00
|
|
|
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)
|
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
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)
|
|
|
|
book_identifier = request.POST.get('book')
|
|
|
|
# TODO: better failure behavior
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
# TODO: validation, htmlification
|
|
|
|
name = form.data.get('name')
|
|
|
|
content = form.data.get('content')
|
2020-04-01 18:22:22 +00:00
|
|
|
rating = form.data.get('rating')
|
2020-02-22 22:02:03 +00:00
|
|
|
|
2020-04-01 14:10:59 +00:00
|
|
|
# throws a value error if the book is not found
|
|
|
|
book = get_or_create_book(book_identifier)
|
|
|
|
|
|
|
|
outgoing.handle_review(request.user, book, name, content, rating)
|
2020-02-22 22:02:03 +00:00
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
|
2020-03-21 23:50:49 +00:00
|
|
|
@login_required
|
|
|
|
def comment(request):
|
|
|
|
''' create a book comment '''
|
|
|
|
form = forms.CommentForm(request.POST)
|
|
|
|
book_identifier = request.POST.get('book')
|
|
|
|
# TODO: better failure behavior
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
# TODO: validation, htmlification
|
|
|
|
content = form.data.get('content')
|
|
|
|
|
2020-04-01 18:22:22 +00:00
|
|
|
outgoing.handle_comment(request.user, book_identifier, content)
|
2020-03-21 23:50:49 +00:00
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
|
2020-02-22 22:02:03 +00:00
|
|
|
@login_required
|
|
|
|
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_identifier = request.POST.get('book')
|
|
|
|
|
|
|
|
outgoing.handle_tag(request.user, book_identifier, name)
|
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def untag(request):
|
|
|
|
''' untag a book '''
|
|
|
|
name = request.POST.get('name')
|
|
|
|
book_identifier = request.POST.get('book')
|
|
|
|
|
|
|
|
outgoing.handle_untag(request.user, book_identifier, name)
|
|
|
|
return redirect('/book/%s' % book_identifier)
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
2020-03-21 23:50:49 +00:00
|
|
|
def reply(request):
|
2020-02-22 22:02:03 +00:00
|
|
|
''' respond to a book review '''
|
2020-03-21 23:50:49 +00:00
|
|
|
form = forms.ReplyForm(request.POST)
|
2020-02-22 22:02:03 +00:00
|
|
|
# this is a bit of a formality, the form is just one text field
|
|
|
|
if not form.is_valid():
|
|
|
|
return redirect('/')
|
|
|
|
parent_id = request.POST['parent']
|
|
|
|
parent = models.Status.objects.get(id=parent_id)
|
2020-03-21 23:50:49 +00:00
|
|
|
outgoing.handle_reply(request.user, parent, form.data['content'])
|
2020-02-22 22:02:03 +00:00
|
|
|
return redirect('/')
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def favorite(request, status_id):
|
|
|
|
''' like a status '''
|
|
|
|
status = models.Status.objects.get(id=status_id)
|
2020-03-29 02:23:19 +00:00
|
|
|
outgoing.handle_favorite(request.user, status)
|
2020-03-21 22:21:27 +00:00
|
|
|
return redirect(request.headers.get('Referer', '/'))
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def unfavorite(request, status_id):
|
|
|
|
''' like a status '''
|
|
|
|
status = models.Status.objects.get(id=status_id)
|
2020-03-29 02:23:19 +00:00
|
|
|
outgoing.handle_unfavorite(request.user, status)
|
2020-02-22 22:02:03 +00:00
|
|
|
return redirect(request.headers.get('Referer', '/'))
|
|
|
|
|
2020-03-30 14:13:32 +00:00
|
|
|
@login_required
|
|
|
|
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
|
|
|
|
|
|
|
@login_required
|
|
|
|
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()
|
|
|
|
|
2020-03-29 02:23:19 +00:00
|
|
|
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
|
|
|
|
def unfollow(request):
|
|
|
|
''' unfollow a user '''
|
|
|
|
username = request.POST['user']
|
|
|
|
try:
|
|
|
|
to_unfollow = get_user_from_username(username)
|
|
|
|
except models.User.DoesNotExist:
|
|
|
|
return HttpResponseBadRequest()
|
|
|
|
|
2020-03-29 02:23:19 +00:00
|
|
|
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)
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
|
|
|
def search(request):
|
|
|
|
''' that search bar up top '''
|
|
|
|
query = request.GET.get('q')
|
|
|
|
if re.match(r'\w+@\w+.\w+', query):
|
|
|
|
# if something looks like a username, search with webfinger
|
|
|
|
results = [outgoing.handle_account_search(query)]
|
|
|
|
template = 'user_results.html'
|
|
|
|
else:
|
2020-03-07 20:22:28 +00:00
|
|
|
# just send the question over to book search
|
|
|
|
results = books_manager.search(query)
|
2020-02-22 22:02:03 +00:00
|
|
|
template = 'book_results.html'
|
|
|
|
|
|
|
|
return TemplateResponse(request, template, {'results': results})
|
|
|
|
|
|
|
|
|
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
|
|
|
|
2020-03-13 14:34:40 +00:00
|
|
|
@login_required
|
|
|
|
def accept_follow_request(request):
|
2020-03-15 21:15:36 +00:00
|
|
|
''' a user accepts a follow request '''
|
2020-03-13 14:34:40 +00:00
|
|
|
username = request.POST['user']
|
|
|
|
try:
|
|
|
|
requester = get_user_from_username(username)
|
|
|
|
except models.User.DoesNotExist:
|
|
|
|
return HttpResponseBadRequest()
|
|
|
|
|
2020-03-13 17:04:39 +00:00
|
|
|
try:
|
2020-03-15 21:15:36 +00:00
|
|
|
follow_request = models.UserFollowRequest.objects.get(
|
|
|
|
user_subject=requester,
|
|
|
|
user_object=request.user
|
|
|
|
)
|
2020-03-13 17:04:39 +00:00
|
|
|
except models.UserFollowRequest.DoesNotExist:
|
|
|
|
# Request already dealt with.
|
|
|
|
pass
|
|
|
|
else:
|
2020-03-29 02:23:19 +00:00
|
|
|
outgoing.handle_accept(requester, request.user, follow_request)
|
2020-03-13 17:04:39 +00:00
|
|
|
|
|
|
|
return redirect('/user/%s' % request.user.localname)
|
2020-03-13 14:34:40 +00:00
|
|
|
|
2020-03-15 21:15:36 +00:00
|
|
|
|
2020-03-13 14:34:40 +00:00
|
|
|
@login_required
|
|
|
|
def delete_follow_request(request):
|
2020-03-15 21:15:36 +00:00
|
|
|
''' a user rejects a follow request '''
|
2020-03-13 14:34:40 +00:00
|
|
|
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
|
|
|
|
)
|
2020-03-13 14:34:40 +00:00
|
|
|
except models.UserFollowRequest.DoesNotExist:
|
2020-03-13 17:04:39 +00:00
|
|
|
return HttpResponseBadRequest()
|
2020-03-13 14:34:40 +00:00
|
|
|
|
2020-03-29 02:23:19 +00:00
|
|
|
outgoing.handle_reject(requester, request.user, follow_request)
|
2020-03-13 14:34:40 +00:00
|
|
|
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
|
|
|
|
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():
|
2020-03-24 16:58:11 +00:00
|
|
|
results = []
|
2020-04-01 14:41:19 +00:00
|
|
|
reviews = []
|
2020-03-24 16:58:11 +00:00
|
|
|
failures = []
|
2020-03-27 16:33:31 +00:00
|
|
|
for item in GoodreadsCsv(TextIOWrapper(
|
|
|
|
request.FILES['csv_file'],
|
|
|
|
encoding=request.encoding)):
|
2020-03-25 12:29:21 +00:00
|
|
|
if item.book:
|
2020-03-25 14:14:38 +00:00
|
|
|
results.append(item)
|
2020-04-01 14:41:19 +00:00
|
|
|
if item.rating:
|
|
|
|
reviews.append(item)
|
2020-03-24 16:58:11 +00:00
|
|
|
else:
|
2020-03-25 12:29:21 +00:00
|
|
|
failures.append(item)
|
2020-03-25 14:14:38 +00:00
|
|
|
|
|
|
|
outgoing.handle_import_books(request.user, results)
|
2020-04-01 14:41:19 +00:00
|
|
|
for item in reviews:
|
|
|
|
outgoing.handle_review(
|
|
|
|
request.user, item.book, "", item.review, item.rating)
|
2020-03-29 07:05:09 +00:00
|
|
|
return TemplateResponse(request, 'import_results.html', {
|
|
|
|
'success_count': len(results),
|
|
|
|
'failures': failures,
|
|
|
|
})
|
2020-04-01 14:18:45 +00:00
|
|
|
return HttpResponseBadRequest()
|