api.py
315 lines
| 9.3 KiB
| text/x-python
|
PythonLexer
neko259
|
r450 | import json | ||
neko259
|
r639 | import logging | ||
neko259
|
r1117 | |||
neko259
|
r1441 | from django.core import serializers | ||
neko259
|
r450 | from django.db import transaction | ||
from django.http import HttpResponse | ||||
neko259
|
r1117 | from django.shortcuts import get_object_or_404 | ||
neko259
|
r1422 | from django.views.decorators.csrf import csrf_protect | ||
neko259
|
r1441 | from boards.abstracts.settingsmanager import get_settings_manager | ||
neko259
|
r533 | from boards.forms import PostForm, PlainErrorList | ||
neko259
|
r1441 | from boards.mdx_neboard import Parser | ||
neko259
|
r1874 | from boards.models import Post, Thread, Tag, Attachment, TagAlias | ||
neko259
|
r1414 | from boards.models.thread import STATUS_ARCHIVE | ||
neko259
|
r1441 | from boards.models.user import Notification | ||
neko259
|
r542 | from boards.utils import datetime_to_epoch | ||
from boards.views.thread import ThreadView | ||||
neko259
|
r1702 | from boards.models.attachment.viewers import FILE_TYPES_IMAGE | ||
neko259
|
r1117 | |||
neko259
|
r450 | __author__ = 'neko259' | ||
neko259
|
r501 | PARAMETER_TRUNCATED = 'truncated' | ||
PARAMETER_TAG = 'tag' | ||||
PARAMETER_OFFSET = 'offset' | ||||
neko259
|
r524 | PARAMETER_DIFF_TYPE = 'type' | ||
neko259
|
r1075 | PARAMETER_POST = 'post' | ||
neko259
|
r1076 | PARAMETER_UPDATED = 'updated' | ||
neko259
|
r1086 | PARAMETER_LAST_UPDATE = 'last_update' | ||
neko259
|
r1119 | PARAMETER_THREAD = 'thread' | ||
PARAMETER_UIDS = 'uids' | ||||
neko259
|
r1625 | PARAMETER_SUBSCRIBED = 'subscribed' | ||
neko259
|
r524 | |||
DIFF_TYPE_HTML = 'html' | ||||
DIFF_TYPE_JSON = 'json' | ||||
neko259
|
r501 | |||
neko259
|
r533 | STATUS_OK = 'ok' | ||
STATUS_ERROR = 'error' | ||||
neko259
|
r639 | logger = logging.getLogger(__name__) | ||
neko259
|
r450 | |||
@transaction.atomic | ||||
neko259
|
r1086 | def api_get_threaddiff(request): | ||
neko259
|
r627 | """ | ||
Gets posts that were changed or added since time | ||||
""" | ||||
neko259
|
r450 | |||
neko259
|
r1191 | thread_id = request.POST.get(PARAMETER_THREAD) | ||
neko259
|
r1425 | uids_str = request.POST.get(PARAMETER_UIDS) | ||
if not thread_id or not uids_str: | ||||
return HttpResponse(content='Invalid request.') | ||||
uids = uids_str.strip().split(' ') | ||||
neko259
|
r639 | |||
neko259
|
r1323 | opening_post = get_object_or_404(Post, id=thread_id) | ||
thread = opening_post.get_thread() | ||||
neko259
|
r450 | |||
json_data = { | ||||
neko259
|
r1076 | PARAMETER_UPDATED: [], | ||
neko259
|
r1119 | PARAMETER_LAST_UPDATE: None, # TODO Maybe this can be removed already? | ||
neko259
|
r504 | } | ||
neko259
|
r1704 | posts = Post.objects.filter(thread=thread).exclude(uid__in=uids) | ||
neko259
|
r524 | |||
neko259
|
r917 | diff_type = request.GET.get(PARAMETER_DIFF_TYPE, DIFF_TYPE_HTML) | ||
neko259
|
r524 | |||
neko259
|
r1118 | for post in posts: | ||
neko259
|
r1340 | json_data[PARAMETER_UPDATED].append(post.get_post_data( | ||
format_type=diff_type, request=request)) | ||||
neko259
|
r1086 | json_data[PARAMETER_LAST_UPDATE] = str(thread.last_edit_time) | ||
neko259
|
r450 | |||
neko259
|
r1625 | settings_manager = get_settings_manager(request) | ||
json_data[PARAMETER_SUBSCRIBED] = str(settings_manager.thread_is_fav(opening_post)) | ||||
neko259
|
r1323 | # If the tag is favorite, update the counter | ||
settings_manager = get_settings_manager(request) | ||||
favorite = settings_manager.thread_is_fav(opening_post) | ||||
if favorite: | ||||
settings_manager.add_or_read_fav_thread(opening_post) | ||||
neko259
|
r450 | return HttpResponse(content=json.dumps(json_data)) | ||
neko259
|
r1422 | @csrf_protect | ||
neko259
|
r533 | def api_add_post(request, opening_post_id): | ||
""" | ||||
neko259
|
r627 | Adds a post and return the JSON response for it | ||
neko259
|
r533 | """ | ||
opening_post = get_object_or_404(Post, id=opening_post_id) | ||||
neko259
|
r640 | logger.info('Adding post via api...') | ||
neko259
|
r533 | status = STATUS_OK | ||
errors = [] | ||||
if request.method == 'POST': | ||||
neko259
|
r640 | form = PostForm(request.POST, request.FILES, error_class=PlainErrorList) | ||
neko259
|
r533 | form.session = request.session | ||
neko259
|
r638 | if form.need_to_ban: | ||
# Ban user because he is suspected to be a bot | ||||
# _ban_current_user(request) | ||||
status = STATUS_ERROR | ||||
neko259
|
r533 | if form.is_valid(): | ||
neko259
|
r640 | post = ThreadView().new_post(request, form, opening_post, | ||
neko259
|
r690 | html_response=False) | ||
neko259
|
r640 | if not post: | ||
neko259
|
r638 | status = STATUS_ERROR | ||
neko259
|
r640 | else: | ||
logger.info('Added post #%d via api.' % post.id) | ||||
neko259
|
r533 | else: | ||
status = STATUS_ERROR | ||||
errors = form.as_json_errors() | ||||
response = { | ||||
'status': status, | ||||
'errors': errors, | ||||
} | ||||
return HttpResponse(content=json.dumps(response)) | ||||
neko259
|
r499 | |||
neko259
|
r491 | def get_post(request, post_id): | ||
neko259
|
r499 | """ | ||
neko259
|
r627 | Gets the html of a post. Used for popups. Post can be truncated if used | ||
neko259
|
r499 | in threads list with 'truncated' get parameter. | ||
""" | ||||
neko259
|
r491 | |||
post = get_object_or_404(Post, id=post_id) | ||||
neko259
|
r1110 | truncated = PARAMETER_TRUNCATED in request.GET | ||
neko259
|
r491 | |||
neko259
|
r1413 | return HttpResponse(content=post.get_view(truncated=truncated, need_op_data=True)) | ||
neko259
|
r499 | |||
def api_get_threads(request, count): | ||||
""" | ||||
neko259
|
r627 | Gets the JSON thread opening posts list. | ||
neko259
|
r499 | Parameters that can be used for filtering: | ||
tag, offset (from which thread to get results) | ||||
""" | ||||
neko259
|
r501 | if PARAMETER_TAG in request.GET: | ||
tag_name = request.GET[PARAMETER_TAG] | ||||
neko259
|
r499 | if tag_name is not None: | ||
tag = get_object_or_404(Tag, name=tag_name) | ||||
neko259
|
r1414 | threads = tag.get_threads().exclude(status=STATUS_ARCHIVE) | ||
neko259
|
r499 | else: | ||
neko259
|
r1414 | threads = Thread.objects.exclude(status=STATUS_ARCHIVE) | ||
neko259
|
r499 | |||
neko259
|
r501 | if PARAMETER_OFFSET in request.GET: | ||
offset = request.GET[PARAMETER_OFFSET] | ||||
neko259
|
r499 | offset = int(offset) if offset is not None else 0 | ||
else: | ||||
offset = 0 | ||||
threads = threads.order_by('-bump_time') | ||||
threads = threads[offset:offset + int(count)] | ||||
opening_posts = [] | ||||
for thread in threads: | ||||
opening_post = thread.get_opening_post() | ||||
neko259
|
r524 | # TODO Add tags, replies and images count | ||
neko259
|
r1340 | post_data = opening_post.get_post_data(include_last_update=True) | ||
neko259
|
r1414 | post_data['status'] = thread.get_status() | ||
neko259
|
r976 | |||
opening_posts.append(post_data) | ||||
neko259
|
r499 | |||
return HttpResponse(content=json.dumps(opening_posts)) | ||||
neko259
|
r500 | |||
neko259
|
r502 | # TODO Test this | ||
neko259
|
r500 | def api_get_tags(request): | ||
""" | ||||
neko259
|
r627 | Gets all tags or user tags. | ||
neko259
|
r500 | """ | ||
# TODO Get favorite tags for the given user ID | ||||
neko259
|
r1874 | tags = TagAlias.objects.all() | ||
neko259
|
r1115 | |||
term = request.GET.get('term') | ||||
if term is not None: | ||||
tags = tags.filter(name__contains=term) | ||||
tag_names = [tag.name for tag in tags] | ||||
neko259
|
r500 | |||
return HttpResponse(content=json.dumps(tag_names)) | ||||
neko259
|
r502 | |||
neko259
|
r1702 | def api_get_stickers(request): | ||
attachments = Attachment.objects.filter(mimetype__in=FILE_TYPES_IMAGE)\ | ||||
.exclude(alias='').exclude(alias=None) | ||||
term = request.GET.get('term') | ||||
if term: | ||||
neko259
|
r1703 | attachments = attachments.filter(alias__contains=term) | ||
neko259
|
r1702 | |||
image_dict = [{'thumb': attachment.get_thumb_url(), | ||||
'alias': attachment.alias} | ||||
for attachment in attachments] | ||||
return HttpResponse(content=json.dumps(image_dict)) | ||||
neko259
|
r502 | # TODO The result can be cached by the thread last update time | ||
# TODO Test this | ||||
def api_get_thread_posts(request, opening_post_id): | ||||
""" | ||||
neko259
|
r627 | Gets the JSON array of thread posts | ||
neko259
|
r502 | """ | ||
opening_post = get_object_or_404(Post, id=opening_post_id) | ||||
neko259
|
r627 | thread = opening_post.get_thread() | ||
neko259
|
r502 | posts = thread.get_replies() | ||
neko259
|
r525 | json_data = { | ||
'posts': [], | ||||
'last_update': None, | ||||
} | ||||
neko259
|
r502 | json_post_list = [] | ||
for post in posts: | ||||
neko259
|
r1340 | json_post_list.append(post.get_post_data()) | ||
neko259
|
r542 | json_data['last_update'] = datetime_to_epoch(thread.last_edit_time) | ||
neko259
|
r525 | json_data['posts'] = json_post_list | ||
neko259
|
r524 | |||
neko259
|
r525 | return HttpResponse(content=json.dumps(json_data)) | ||
neko259
|
r524 | |||
neko259
|
r994 | def api_get_notifications(request, username): | ||
last_notification_id_str = request.GET.get('last', None) | ||||
last_id = int(last_notification_id_str) if last_notification_id_str is not None else None | ||||
neko259
|
r1445 | posts = Notification.objects.get_notification_posts(usernames=[username], | ||
neko259
|
r1117 | last=last_id) | ||
neko259
|
r994 | |||
json_post_list = [] | ||||
for post in posts: | ||||
neko259
|
r1340 | json_post_list.append(post.get_post_data()) | ||
neko259
|
r994 | return HttpResponse(content=json.dumps(json_post_list)) | ||
neko259
|
r564 | def api_get_post(request, post_id): | ||
""" | ||||
neko259
|
r627 | Gets the JSON of a post. This can be | ||
neko259
|
r564 | used as and API for external clients. | ||
""" | ||||
post = get_object_or_404(Post, id=post_id) | ||||
json = serializers.serialize("json", [post], fields=( | ||||
"pub_time", "_text_rendered", "title", "text", "image", | ||||
"image_width", "image_height", "replies", "tags" | ||||
)) | ||||
return HttpResponse(content=json) | ||||
neko259
|
r1217 | def api_get_preview(request): | ||
raw_text = request.POST['raw_text'] | ||||
parser = Parser() | ||||
return HttpResponse(content=parser.parse(parser.preparse(raw_text))) | ||||
neko259
|
r1340 | |||
def api_get_new_posts(request): | ||||
""" | ||||
Gets favorite threads and unread posts count. | ||||
""" | ||||
posts = list() | ||||
include_posts = 'include_posts' in request.GET | ||||
settings_manager = get_settings_manager(request) | ||||
fav_threads = settings_manager.get_fav_threads() | ||||
fav_thread_ops = Post.objects.filter(id__in=fav_threads.keys())\ | ||||
.order_by('-pub_time').prefetch_related('thread') | ||||
neko259
|
r1345 | ops = [{'op': op, 'last_id': fav_threads[str(op.id)]} for op in fav_thread_ops] | ||
if include_posts: | ||||
new_post_threads = Thread.objects.get_new_posts(ops) | ||||
neko259
|
r1346 | if new_post_threads: | ||
thread_ids = {thread.id: thread for thread in new_post_threads} | ||||
else: | ||||
thread_ids = dict() | ||||
neko259
|
r1345 | |||
for op in fav_thread_ops: | ||||
fav_thread_dict = dict() | ||||
neko259
|
r1344 | |||
neko259
|
r1346 | op_thread = op.get_thread() | ||
if op_thread.id in thread_ids: | ||||
thread = thread_ids[op_thread.id] | ||||
neko259
|
r1345 | new_post_count = thread.new_post_count | ||
neko259
|
r1346 | fav_thread_dict['newest_post_link'] = thread.get_replies()\ | ||
.filter(id__gt=fav_threads[str(op.id)])\ | ||||
neko259
|
r1365 | .first().get_absolute_url(thread=thread) | ||
neko259
|
r1345 | else: | ||
new_post_count = 0 | ||||
fav_thread_dict['new_post_count'] = new_post_count | ||||
neko259
|
r1344 | |||
neko259
|
r1345 | fav_thread_dict['id'] = op.id | ||
neko259
|
r1340 | |||
neko259
|
r1343 | fav_thread_dict['post_url'] = op.get_link_view() | ||
fav_thread_dict['title'] = op.title | ||||
neko259
|
r1340 | |||
neko259
|
r1345 | posts.append(fav_thread_dict) | ||
else: | ||||
fav_thread_dict = dict() | ||||
fav_thread_dict['new_post_count'] = \ | ||||
Thread.objects.get_new_post_count(ops) | ||||
neko259
|
r1340 | posts.append(fav_thread_dict) | ||
return HttpResponse(content=json.dumps(posts)) | ||||