changeset 810:4a4fa174a0ec

Private message refactor: adding ability to report PM's. See also Bitbucket issue #57.
author Brian Neal <bgneal@gmail.com>
date Sat, 06 Sep 2014 16:58:08 -0500 (2014-09-06)
parents ab3deff7672a
children 56b30c79f10e
files core/templatetags/custom_admin_tags.py messages/admin.py messages/forms.py messages/models.py messages/static/css/messages.css messages/urls.py messages/views.py sg101/templates/admin/base_site.html sg101/templates/core/admin_dashboard.html sg101/templates/messages/message.html sg101/templates/messages/report_message.html sg101/templates/messages/view_message.html
diffstat 12 files changed, 180 insertions(+), 28 deletions(-) [+]
line wrap: on
line diff
--- a/core/templatetags/custom_admin_tags.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/core/templatetags/custom_admin_tags.py	Sat Sep 06 16:58:08 2014 -0500
@@ -12,6 +12,7 @@
 from news.models import PendingStory
 from weblinks.models import PendingLink, FlaggedLink
 from shoutbox.models import ShoutFlag
+from messages.models import Flag as PmFlag
 
 
 register = template.Library()
@@ -28,13 +29,14 @@
     new_downloads = PendingDownload.objects.count()
     flagged_posts = FlaggedPost.objects.count()
     event_requests = Event.objects.filter(
-                Q(status=Event.NEW) | 
-                Q(status=Event.EDIT_REQ) | 
+                Q(status=Event.NEW) |
+                Q(status=Event.EDIT_REQ) |
                 Q(status=Event.DEL_REQ)).count()
     new_stories = PendingStory.objects.count()
     new_links = PendingLink.objects.count()
     broken_links = FlaggedLink.objects.count()
     flagged_shouts = ShoutFlag.objects.count()
+    flagged_msgs = PmFlag.objects.count()
 
     return {
         'user': user,
@@ -47,4 +49,5 @@
         'new_links': new_links,
         'broken_links': broken_links,
         'flagged_shouts': flagged_shouts,
+        'flagged_msgs': flagged_msgs,
         }
--- a/messages/admin.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/admin.py	Sat Sep 06 16:58:08 2014 -0500
@@ -2,17 +2,53 @@
 This file contains the automatic admin site definitions for the Message models.
 """
 from django.contrib import admin
+from django.core.urlresolvers import reverse
 
-from messages.models import Message
-from messages.models import Options
+from messages.models import Flag, Message, Options
+import bio.badges
 
 
 class MessageAdmin(admin.ModelAdmin):
-    list_display = ('sender', 'receiver', 'send_date', 'subject')
-    raw_id_fields = ('sender', 'receiver')
-    exclude = ('html', )
+    list_display = ['sender', 'receiver', 'send_date', 'subject']
+    raw_id_fields = ['sender', 'receiver']
+    exclude = ['html']
     date_hierarchy = 'send_date'
-    list_display_links = ('subject', )
+    list_display_links = ['subject']
 
+
+class FlagAdmin(admin.ModelAdmin):
+    list_display = ['__unicode__', 'flag_date', 'message_link']
+    date_hierarchy = 'flag_date'
+    list_select_related = True
+    readonly_fields = ['message', 'flag_date', 'comments', 'message_text']
+    actions = ['accept_flags']
+
+    def message_link(self, obj):
+        return '<a href="{}">Message</a>'.format(
+                reverse('admin:messages_message_change', args=[obj.message.id]))
+    message_link.allow_tags = True
+    message_link.short_description = 'Message'
+
+    def message_text(self, obj):
+        return obj.message.html
+    message_text.allow_tags = True
+    message_text.short_description = 'Message text'
+
+    def accept_flags(self, request, qs):
+        """This admin action awards a security pin to the user who reported the
+        message and then deletes the flag object.
+
+        """
+        count = qs.count()
+        for flag in qs:
+            bio.badges.award_badge(bio.badges.SECURITY_PIN, flag.message.receiver)
+            flag.delete()
+
+        self.message_user(request,
+            "%s flag(s) acknowledged. You may want to mark users as spammers now." % count)
+
+    accept_flags.short_description = "Accept selected flags"
+
+admin.site.register(Flag, FlagAdmin)
 admin.site.register(Message, MessageAdmin)
 admin.site.register(Options)
--- a/messages/forms.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/forms.py	Sat Sep 06 16:58:08 2014 -0500
@@ -11,8 +11,7 @@
 
 from core.functions import send_mail
 from core.widgets import AutoCompleteUserInput
-from messages.models import Message
-from messages.models import Options
+from messages.models import Flag, Message, Options
 from messages import MSG_BOX_LIMIT
 
 
@@ -124,6 +123,17 @@
         fields = ['attach_signature', 'notify_email']
 
 
+class ReportForm(forms.ModelForm):
+    class Meta:
+        model = Flag
+        fields = ['comments']
+        labels = {'comments': ''}
+        widgets = {
+            'comments': forms.Textarea(attrs={
+                'placeholder': 'Enter a comment for the admin (optional).'})
+        }
+
+
 def notify_receiver(new_msg):
     """
     This function creates the notification email to notify a user of
--- a/messages/models.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/models.py	Sat Sep 06 16:58:08 2014 -0500
@@ -101,3 +101,22 @@
         verbose_name_plural = "Options"
 
 
+class Flag(models.Model):
+    """The Flag model is used to represent when a receiver of a private message
+    has flagged the message for spam or abuse.
+
+    """
+    message = models.OneToOneField(Message)
+    flag_date = models.DateTimeField()
+    comments = models.TextField(blank=True)
+
+    def __unicode__(self):
+        msg = self.message
+        return "{} has flagged a PM from {}".format(msg.receiver.username,
+                msg.sender.username)
+
+    def save(self, *args, **kwargs):
+        if not self.id:
+            self.flag_date = datetime.datetime.now()
+        super(Flag, self).save(*args, **kwargs)
+
--- a/messages/static/css/messages.css	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/static/css/messages.css	Sat Sep 06 16:58:08 2014 -0500
@@ -90,3 +90,9 @@
 table.pm td {
    background-color: #EDF7F6;
 }
+#report_form textarea {
+   width: 90%;
+   margin-left: auto;
+   margin-right: auto;
+   height: 10em;
+}
--- a/messages/urls.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/urls.py	Sat Sep 06 16:58:08 2014 -0500
@@ -30,7 +30,7 @@
     url(r'^view/(\d+)/$',
         'messages.views.view',
         name='messages-view'),
-#   url(r'^report/(\d+)/$',
-#       'messages.views.report',
-#       name='messages-report'),
+    url(r'^report/(\d+)/$',
+        'messages.views.report',
+        name='messages-report'),
 )
--- a/messages/views.py	Mon Sep 01 17:05:30 2014 -0500
+++ b/messages/views.py	Sat Sep 06 16:58:08 2014 -0500
@@ -12,8 +12,8 @@
 from django.shortcuts import get_object_or_404
 from django.shortcuts import render, redirect
 
-from messages.models import Message, Options
-from messages.forms import OptionsForm, ComposeForm
+from messages.models import Flag, Message, Options
+from messages.forms import OptionsForm, ComposeForm, ReportForm
 from messages.utils import reply_subject
 from messages import MSG_BOX_LIMIT
 from core.functions import quote_message
@@ -184,9 +184,15 @@
 
         form = ComposeForm(request.user, initial=initial_data)
 
+    try:
+        msg_flag = msg.flag
+    except Flag.DoesNotExist:
+        msg_flag = None
+
     return render(request, 'messages/view_message.html', {
         'msg': msg,
         'form': form,
+        'msg_flag': msg_flag,
         })
 
 
@@ -295,3 +301,38 @@
 
     url = reverse('messages-trash') + '?page={}'.format(page)
     return redirect(url)
+
+
+@login_required
+def report(request, msg_id):
+    """This view is for reporting a PM as spam or abuse.
+
+    """
+    msg = get_object_or_404(Message.objects.select_related(), pk=msg_id)
+    if msg.receiver != request.user:
+        django_messages.error(request, "You can't report this message.")
+        return redirect('messages-inbox')
+    try:
+        msg.flag
+    except Flag.DoesNotExist:
+        pass
+    else:
+        django_messages.error(request, "This message has already been reported.")
+        return redirect('messages-inbox')
+
+    if request.method == 'POST':
+        form = ReportForm(request.POST)
+        if form.is_valid():
+            flag = form.save(commit=False)
+            flag.message = msg
+            flag.save()
+            django_messages.success(request,
+                    'Message reported. An admin will be notified. Thank you.')
+            return redirect('messages-inbox')
+    else:
+        form = ReportForm()
+
+    return render(request, 'messages/report_message.html', {
+        'msg': msg,
+        'form': form,
+        })
--- a/sg101/templates/admin/base_site.html	Mon Sep 01 17:05:30 2014 -0500
+++ b/sg101/templates/admin/base_site.html	Sat Sep 06 16:58:08 2014 -0500
@@ -9,9 +9,8 @@
    }
    #dashboard-list li {
       float: left;
+      display: inline;
       list-style: square inside none;
-      margin-right: 1.5em;
-      margin-bottom: 0.5em;
       margin: 0.5em 1.0em 0.5em 1.5em;
    }
 </style>
--- a/sg101/templates/core/admin_dashboard.html	Mon Sep 01 17:05:30 2014 -0500
+++ b/sg101/templates/core/admin_dashboard.html	Sat Sep 06 16:58:08 2014 -0500
@@ -1,5 +1,5 @@
 {% if user.is_staff %}
-{% if flagged_posts or flagged_comments or flagged_profiles or event_requests or new_stories or new_downloads or new_links or flagged_shouts or broken_links %}
+{% if flagged_posts or flagged_comments or flagged_profiles or event_requests or new_stories or new_downloads or new_links or flagged_shouts or broken_links or flagged_msgs %}
 <ul id="dashboard-list">
 {% if flagged_posts %}
 <li><a href="/admin/forums/flaggedpost/">Posts</a>: {{ flagged_posts }}</li>
@@ -28,6 +28,9 @@
 {% if broken_links %}
 <li><a href="/admin/weblinks/flaggedlink/">Broken Links</a>: {{ broken_links }}</li>
 {% endif %}
+{% if flagged_msgs %}
+<li><a href="/admin/messages/flag/">PM's</a>: {{ flagged_msgs }}</li>
+{% endif %}
 </ul>
 {% endif %}
 {% endif %}
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/sg101/templates/messages/message.html	Sat Sep 06 16:58:08 2014 -0500
@@ -0,0 +1,10 @@
+{% load bio_tags %}
+<table class="pm">
+   <tr><th>From:</th><td>{% profile_link msg.sender.username %}</td></tr>
+   <tr><th>To:</th><td>{% profile_link msg.receiver.username %}</td></tr>
+   <tr><th>Send Date:</th><td>{{ msg.send_date }}</td></tr>
+   <tr><th>Read Date:</th><td>{{ msg.read_date }}</td></tr>
+   <tr><th>Reply Date:</th><td>{{ msg.reply_date }}</td></tr>
+   <tr><th>Subject:</th><td>{{ msg.subject }}</td></tr>
+   <tr><td colspan="2">{{ msg.html|safe }}</td></tr>
+</table>
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/sg101/templates/messages/report_message.html	Sat Sep 06 16:58:08 2014 -0500
@@ -0,0 +1,23 @@
+{% extends 'messages/messages_base.html' %}
+{% block messages_content %}
+<h3>Report Private Message</h3>
+<p>
+You have asked to report this private message for spam or abuse. If you wish to
+proceed, please submit the form below. You may also add an optional note or clarification
+for the admin.
+</p>
+<p>
+If you submit the form an admin will be notified and the message will be
+reviewed and possibly deleted.
+</p>
+{% include "messages/message.html" %}
+
+<form action="." method="post" id="report_form">{% csrf_token %}
+<fieldset>
+<legend>Report Message</legend>
+{{ form.as_p }}
+<input type="submit" name="submit" value="Submit Report" />
+</fieldset>
+</form>
+<p><a href="{% url "messages-view" msg.id %}">Cancel report and return to message</a></p>
+{% endblock %}
--- a/sg101/templates/messages/view_message.html	Mon Sep 01 17:05:30 2014 -0500
+++ b/sg101/templates/messages/view_message.html	Sat Sep 06 16:58:08 2014 -0500
@@ -1,5 +1,4 @@
 {% extends 'messages/messages_base.html' %}
-{% load bio_tags %}
 {% load core_tags %}
 {% load script_tags %}
 {% block custom_js %}
@@ -10,15 +9,18 @@
 {% endblock %}
 {% block messages_content %}
 <h3>View Private Message</h3>
-<table class="pm">
-   <tr><th>From:</th><td>{% profile_link msg.sender.username %}</td></tr>
-   <tr><th>To:</th><td>{% profile_link msg.receiver.username %}</td></tr>
-   <tr><th>Send Date:</th><td>{{ msg.send_date }}</td></tr>
-   <tr><th>Read Date:</th><td>{{ msg.read_date }}</td></tr>
-   <tr><th>Reply Date:</th><td>{{ msg.reply_date }}</td></tr>
-   <tr><th>Subject:</th><td>{{ msg.subject }}</td></tr>
-   <tr><td colspan="2">{{ msg.html|safe }}</td></tr>
-</table>
+{% include "messages/message.html" %}
+
+{% if msg_flag %}
+<div class="error">
+   You reported this message to the admin on {{ msg_flag.flag_date|date:"F d, Y" }}.
+</div>
+{% else %}
+<div class="notice">
+   <img src="{{ STATIC_URL }}icons/flag_red.png" alt="Report this message" />
+   <a href="{% url "messages-report" msg.id %}">Report this message for spam or abuse</a>.
+</div>
+{% endif %}
 
 {% if form %}
 <form action="." method="post" id="msg_compose_form">{% csrf_token %}