From 4483cf6166cacf7d54335be4188af9810e6d12b4 Mon Sep 17 00:00:00 2001
From: Carlos Vega <carlos.vega@uni.lu>
Date: Mon, 9 Nov 2020 17:06:29 +0100
Subject: [PATCH] added dynamic table that loads via AJAX. This should support
 thousands of rows without issue

---
 requirements.txt                         |  1 +
 smash/web/api_urls.py                    |  4 +-
 smash/web/api_views/provenance.py        | 86 ++++++++++++++++++++++++
 smash/web/templates/export/index.html    | 58 +++++++++++-----
 smash/web/templates/provenance/list.html | 83 +++++++++++++++++------
 smash/web/utils.py                       |  7 ++
 6 files changed, 202 insertions(+), 37 deletions(-)
 create mode 100644 smash/web/api_views/provenance.py

diff --git a/requirements.txt b/requirements.txt
index 343e3de8..36a5df54 100644
--- a/requirements.txt
+++ b/requirements.txt
@@ -33,6 +33,7 @@ mockito==1.2.2
 nexmo==2.3.0
 numpy==1.16.1
 pandas==1.1.3
+django-datatables-view==1.19.1
 phonenumberslite==8.9.14
 Pillow==3.4.2
 psycopg2==2.8.6
diff --git a/smash/web/api_urls.py b/smash/web/api_urls.py
index 22f3d57d..7dae498a 100644
--- a/smash/web/api_urls.py
+++ b/smash/web/api_urls.py
@@ -16,7 +16,7 @@ Including another URLconf
 from django.conf.urls import url
 
 from web.api_views import worker, location, subject, appointment_type, appointment, configuration, daily_planning, \
-    redcap, flying_team, visit, voucher, voucher_type
+    redcap, flying_team, visit, voucher, voucher_type, provenance
 
 urlpatterns = [
     # appointments
@@ -35,6 +35,8 @@ urlpatterns = [
     # subjects data
     url(r'^cities$', subject.cities, name='web.api.cities'),
     url(r'^referrals$', subject.referrals, name='web.api.referrals'),
+    url(r'^export_log$', provenance.ExportLog.as_view(), name='web.api.export_log'),
+    url(r'^provenances$', provenance.CompleteLog.as_view(), name='web.api.provenances'),
     url(r'^subjects/(?P<type>[A-z]+)$', subject.subjects, name='web.api.subjects'),
     url(r'^subjects:columns/(?P<subject_list_type>[A-z]+)$', subject.get_subject_columns,
         name='web.api.subjects.columns'),
diff --git a/smash/web/api_views/provenance.py b/smash/web/api_views/provenance.py
new file mode 100644
index 00000000..6fe1407d
--- /dev/null
+++ b/smash/web/api_views/provenance.py
@@ -0,0 +1,86 @@
+from ..models import Provenance, Worker
+from django_datatables_view.base_datatable_view import BaseDatatableView
+
+class ExportLog(BaseDatatableView):
+    model = Provenance
+    columns = [
+        'modification_date',
+        'modification_author',
+        'modification_description',
+        'request_path',
+        'request_ip_addr'
+    ]
+
+    # define column names that will be used in sorting
+    # order is important and should be same as order of columns
+    # displayed by datatables. For non sortable columns use empty
+    # value like ''
+    order_columns = [
+        'modification_date',
+        'modification_author',
+        'modification_description',
+        'request_path',
+        'request_ip_addr'
+    ]
+    
+    # set max limit of records returned, this is used to protect our site if someone tries to attack our site
+    # and make it return huge amount of data
+    max_display_length = 10
+
+    def filter_queryset(self, qs):
+        # simple example:
+        search = self.request.GET.get('search[value]', None)
+        if search != '' and search is not None:
+            inner_qs = Worker.objects.filter(first_name__icontains=search) | Worker.objects.filter(last_name__icontains=search)
+            qs = qs.filter(modification_author__in=inner_qs) | qs.filter(modification_description__icontains=search) | qs.filter(request_path__icontains=search) | qs.filter(request_ip_addr__icontains=search)
+        
+        qs = qs & qs.filter(modified_table__isnull=True, 
+                            previous_value__isnull=True, 
+                            new_value__isnull=True, 
+                            modified_field='', 
+                            request_path__startswith='/export/')
+
+        return qs
+
+class CompleteLog(BaseDatatableView):
+    model = Provenance
+    columns = ['modified_table',
+        'modified_table_id',
+        'modification_date',
+        'modification_author',
+        'modified_field',
+        'previous_value',
+        'new_value',
+        'modification_description',
+        'request_path',
+        'request_ip_addr'
+    ]
+
+    # define column names that will be used in sorting
+    # order is important and should be same as order of columns
+    # displayed by datatables. For non sortable columns use empty
+    # value like ''
+    order_columns = ['modified_table',
+        'modified_table_id',
+        'modification_date',
+        'modification_author',
+        'modified_field',
+        'previous_value',
+        'new_value',
+        'modification_description',
+        'request_path',
+        'request_ip_addr'
+    ]
+    
+    # set max limit of records returned, this is used to protect our site if someone tries to attack our site
+    # and make it return huge amount of data
+    max_display_length = 10
+
+    def filter_queryset(self, qs):
+        # simple example:
+        search = self.request.GET.get('search[value]', None)
+        if search != '' and search is not None:
+            inner_qs = Worker.objects.filter(first_name__icontains=search) | Worker.objects.filter(last_name__icontains=search)
+            qs = qs.filter(modification_author__in=inner_qs) | qs.filter(modification_description__icontains=search) | qs.filter(request_path__icontains=search) | qs.filter(request_ip_addr__icontains=search)
+
+        return qs
\ No newline at end of file
diff --git a/smash/web/templates/export/index.html b/smash/web/templates/export/index.html
index 112fe330..4e99d70b 100644
--- a/smash/web/templates/export/index.html
+++ b/smash/web/templates/export/index.html
@@ -105,20 +105,15 @@
                     <th>Author</th>
                     <th>Description</th>
                     <th>Request Path</th>
+                    <th>Request IP</th>
                 </tr>
                 </thead>
                 <tbody>
-                {% for provenance in provenances %}
-                    <tr>
-                        <td data-sort="{{ provenance.modification_date | timestamp }}">{{ provenance.modification_date }}</td>
-                        <td>{{ provenance.modification_author }}</td>
-                        <td>{{ provenance.modification_description }}</td>
-                        <td>{{ provenance.request_path }}</td>
-                    </tr>
-                {% endfor %}
+
                 </tbody>
             </table>
         </div>
+
     </div>
 
     <div class="box-body">
@@ -139,6 +134,9 @@
                 $(e.target).children('i.fa').toggleClass('fa-expand');
                 $(e.target).children('i.fa').toggleClass('fa-compress');
             });
+
+
+
         };
         function addFields(e, cls){
             var fields = $(`.${cls} > li > input:checked`).map(function(i,e){
@@ -161,14 +159,42 @@
         }
 
         $(function () {
-            $('#table').DataTable({
-                "paging": true,
-                "lengthChange": false,
-                "searching": true,
-                "ordering": true,
-                "order": [[ 0, "desc" ]],
-                "info": true,
-                "autoWidth": false
+            console.log("{% url 'web.api.provenances' %}")
+            var oTable = $('#table').DataTable({
+                processing: true,
+                serverSide: true,
+                ordering: true,
+                autoWidth: false,
+                lengthChange: false,
+                columns: [
+                    {
+                        data: 'modification_date',
+                        orderable: true,
+                        searchable: true
+                    },
+                    {
+                        data: 'modification_author',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'modification_description',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'request_path',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'request_ip_addr',
+                        orderable: true,
+                        searchable: true,
+                    }
+                ],
+                order: [[ 0, "desc" ]],
+                ajax: "{% url 'web.api.export_log' %}"
             });
         });
     </script>
diff --git a/smash/web/templates/provenance/list.html b/smash/web/templates/provenance/list.html
index 44910e46..34051837 100644
--- a/smash/web/templates/provenance/list.html
+++ b/smash/web/templates/provenance/list.html
@@ -31,22 +31,10 @@
                 <th>New Value</th>
                 <th>Description</th>
                 <th>Request Path</th>
+                <th>Request IP</th>
             </tr>
             </thead>
             <tbody>
-            {% for provenance in provenances %}
-                <tr>
-                    <td>{{ provenance.modified_table }}</td>
-                    <td>{{ provenance.modified_table_id }}</td>
-                    <td data-sort="{{ provenance.modification_date | timestamp }}">{{ provenance.modification_date }}</td>
-                    <td>{{ provenance.modification_author }}</td>
-                    <td>{{ provenance.modified_field }}</td>
-                    <td>{{ provenance.previous_value }}</td>
-                    <td>{{ provenance.new_value }}</td>
-                    <td>{{ provenance.modification_description }}</td>
-                    <td>{{ provenance.request_path }}</td>
-                </tr>
-            {% endfor %}
             </tbody>
         </table>
     </div>
@@ -61,13 +49,68 @@
     <script>
         $(function () {
             $('#table').DataTable({
-                "paging": true,
-                "lengthChange": false,
-                "searching": true,
-                "ordering": true,
-                "order": [[ 2, "desc" ]],
-                "info": true,
-                "autoWidth": false
+                paging: true,
+                lengthChange: false,
+                searching: true,
+                processing: true,
+                serverSide: true,
+                ordering: true,
+                order: [[ 2, "desc" ]],
+                info: true,
+                autoWidth: false,
+                columns: [
+                    {
+                        data: 'modified_table',
+                        orderable: true,
+                        searchable: true
+                    },
+                    {
+                        data: 'modified_table_id',
+                        orderable: true,
+                        searchable: true
+                    },
+                    {
+                        data: 'modification_date',
+                        orderable: true,
+                        searchable: true
+                    },
+                    {
+                        data: 'modification_author',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'modified_field',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'previous_value',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'new_value',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'modification_description',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'request_path',
+                        orderable: true,
+                        searchable: true,
+                    },
+                    {
+                        data: 'request_ip_addr',
+                        orderable: true,
+                        searchable: true,
+                    }
+                ],
+                ajax: "{% url 'web.api.provenances' %}"
             });
         });
     </script>
diff --git a/smash/web/utils.py b/smash/web/utils.py
index dd61c944..54666468 100644
--- a/smash/web/utils.py
+++ b/smash/web/utils.py
@@ -5,6 +5,13 @@ from datetime import timedelta
 
 from web.algorithm import VerhoeffAlgorithm, LuhnAlgorithm
 
+def get_client_ip(request):
+    x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
+    if x_forwarded_for:
+        ip = x_forwarded_for.split(',')[0]
+    else:
+        ip = request.META.get('REMOTE_ADDR')
+    return ip
 
 def is_valid_social_security_number(number):
     if number is not None and number != '':
-- 
GitLab