Improve statistics subclassing. Split class module into classes and renderers.

Signed-off-by: Roberto Rosario <roberto.rosario.gonzalez@gmail.com>
This commit is contained in:
Roberto Rosario
2018-08-23 13:05:34 -04:00
parent d36faa5b7a
commit d5b9c785fb
7 changed files with 105 additions and 81 deletions

View File

@@ -106,6 +106,8 @@
- After queuing a chart for update, the view will now redirect
to the same chart.
- The multiple document action dropdown is now sorted alphabetically.
- Improve statistics subclassing. Split class module into classes
and renderers.
3.0.3 (2018-08-17)
==================

View File

@@ -8,7 +8,7 @@ from django.utils.translation import ugettext_lazy as _
import qsstats
from mayan_statistics.classes import StatisticNamespace, CharJSLine
from mayan_statistics import StatisticLineChart, StatisticNamespace
from .permissions import permission_document_view
@@ -230,44 +230,44 @@ def total_document_page_per_month():
namespace = StatisticNamespace(slug='documents', label=_('Documents'))
namespace.add_statistic(
klass=StatisticLineChart,
slug='new-documents-per-month',
label=_('New documents per month'),
func=new_documents_per_month,
renderer=CharJSLine,
minute='0'
)
namespace.add_statistic(
klass=StatisticLineChart,
slug='new-document-versions-per-month',
label=_('New document versions per month'),
func=new_document_versions_per_month,
renderer=CharJSLine,
minute='0'
)
namespace.add_statistic(
klass=StatisticLineChart,
slug='new-document-pages-per-month',
label=_('New document pages per month'),
func=new_document_pages_per_month,
renderer=CharJSLine,
minute='0'
)
namespace.add_statistic(
klass=StatisticLineChart,
slug='total-documents-at-each-month',
label=_('Total documents at each month'),
func=total_document_per_month,
renderer=CharJSLine,
minute='0'
)
namespace.add_statistic(
klass=StatisticLineChart,
slug='total-document-versions-at-each-month',
label=_('Total document versions at each month'),
func=total_document_version_per_month,
renderer=CharJSLine,
minute='0'
)
namespace.add_statistic(
klass=StatisticLineChart,
slug='total-document-pages-at-each-month',
label=_('Total document pages at each month'),
func=total_document_page_per_month,
renderer=CharJSLine,
minute='0'
)

View File

@@ -1,3 +1,5 @@
from __future__ import unicode_literals
from .classes import StatisticLineChart, StatisticNamespace
default_app_config = 'mayan_statistics.apps.StatisticsApp'

View File

@@ -9,7 +9,7 @@ from common import MayanAppConfig, menu_object, menu_secondary, menu_tools
from navigation import SourceColumn
from .classes import Statistic, StatisticNamespace
from .classes import Statistic, StatisticLineChart, StatisticNamespace
from .links import (
link_execute, link_namespace_details, link_namespace_list,
link_statistics, link_view
@@ -29,9 +29,9 @@ class StatisticsApp(MayanAppConfig):
super(StatisticsApp, self).ready()
SourceColumn(
source=Statistic,
# Translators: Schedule here is a verb, the 'schedule' at which the
# statistic will be updated
source=StatisticLineChart,
# Translators: Schedule here is a verb, the 'schedule' at
# which the statistic will be updated
label=_('Schedule'),
attribute='schedule',
)
@@ -54,7 +54,7 @@ class StatisticsApp(MayanAppConfig):
)
menu_object.bind_links(
links=(link_execute, link_view), sources=(Statistic,)
links=(link_execute, link_view), sources=(StatisticLineChart,)
)
menu_object.bind_links(
links=(link_namespace_details,), sources=(StatisticNamespace,)

View File

@@ -9,6 +9,8 @@ from celery.schedules import crontab
from mayan.celery import app
from .renderers import ChartJSLine
@python_2_unicode_compatible
class StatisticNamespace(object):
@@ -31,8 +33,8 @@ class StatisticNamespace(object):
def __str__(self):
return force_text(self.label)
def add_statistic(self, *args, **kwargs):
statistic = Statistic(*args, **kwargs)
def add_statistic(self, klass, *args, **kwargs):
statistic = klass(*args, **kwargs)
statistic.namespace = self
self._statistics.append(statistic)
@@ -44,6 +46,7 @@ class StatisticNamespace(object):
@python_2_unicode_compatible
class Statistic(object):
_registry = {}
renderer = None
@staticmethod
def purge_schedules():
@@ -82,11 +85,10 @@ class Statistic(object):
def get_task_names(cls):
return [task.get_task_name() for task in cls.get_all()]
def __init__(self, slug, label, func, renderer, minute='*', hour='*', day_of_week='*', day_of_month='*', month_of_year='*'):
def __init__(self, slug, label, func, minute='*', hour='*', day_of_week='*', day_of_month='*', month_of_year='*'):
self.slug = slug
self.label = label
self.func = func
self.renderer = renderer
self.schedule = crontab(
minute=minute, hour=hour, day_of_week=day_of_week,
@@ -123,7 +125,9 @@ class Statistic(object):
return 'mayan_statistics.task_execute_statistic_{}'.format(self.slug)
def store_results(self, results):
from .models import StatisticResult
StatisticResult = apps.get_model(
app_label='mayan_statistics', model_name='StatisticResult'
)
StatisticResult.objects.filter(slug=self.slug).delete()
@@ -133,7 +137,9 @@ class Statistic(object):
statistic_result.store_data(data=results)
def get_results(self):
from .models import StatisticResult
StatisticResult = apps.get_model(
app_label='mayan_statistics', model_name='StatisticResult'
)
try:
return StatisticResult.objects.get(slug=self.slug).get_data()
@@ -144,66 +150,5 @@ class Statistic(object):
return self.renderer(data=self.get_results()).get_chart_data()
class ChartRenderer(object):
def __init__(self, data):
self.data = data
def get_chart_data(self):
raise NotImplementedError
class CharJSLine(ChartRenderer):
template_name = 'statistics/backends/chartjs/line.html'
dataset_palette = (
{
'fillColor': "rgba(220,220,220,0.2)",
'strokeColor': "rgba(220,220,220,1)",
'pointColor': "rgba(220,220,220,1)",
'pointStrokeColor': "#fff",
'pointHighlightFill': "#fff",
'pointHighlightStroke': "rgba(220,220,220,1)",
},
{
'fillColor': "rgba(151,187,205,0.2)",
'strokeColor': "rgba(151,187,205,1)",
'pointColor': "rgba(151,187,205,1)",
'pointStrokeColor': "#fff",
'pointHighlightFill': "#fff",
'pointHighlightStroke': "rgba(151,187,205,1)",
},
)
def get_chart_data(self):
labels = []
datasets = []
for count, serie in enumerate(self.data['series'].items()):
series_name, series_data = serie
dataset_labels = []
dataset_values = []
for data_point in series_data:
dataset_labels.extend(data_point.keys())
dataset_values.extend(data_point.values())
labels = dataset_labels
dataset = {
'label': series_name,
'data': dataset_values,
}
dataset.update(
CharJSLine.dataset_palette[
count % len(CharJSLine.dataset_palette)
]
)
datasets.append(dataset)
data = {
'labels': labels,
'datasets': datasets,
}
return json.dumps(data)
class StatisticLineChart(Statistic):
renderer = ChartJSLine

View File

@@ -0,0 +1,75 @@
from __future__ import unicode_literals
import json
from django.apps import apps
from django.utils.encoding import force_text, python_2_unicode_compatible
from celery.schedules import crontab
from mayan.celery import app
class ChartRenderer(object):
def __init__(self, data):
self.data = data
def get_chart_data(self):
raise NotImplementedError
class ChartJSLine(ChartRenderer):
template_name = 'statistics/renderers/chartjs/line.html'
dataset_palette = (
{
'fillColor': "rgba(220,220,220,0.2)",
'strokeColor': "rgba(220,220,220,1)",
'pointColor': "rgba(220,220,220,1)",
'pointStrokeColor': "#fff",
'pointHighlightFill': "#fff",
'pointHighlightStroke': "rgba(220,220,220,1)",
},
{
'fillColor': "rgba(151,187,205,0.2)",
'strokeColor': "rgba(151,187,205,1)",
'pointColor': "rgba(151,187,205,1)",
'pointStrokeColor': "#fff",
'pointHighlightFill': "#fff",
'pointHighlightStroke': "rgba(151,187,205,1)",
},
)
def get_chart_data(self):
labels = []
datasets = []
for count, serie in enumerate(self.data['series'].items()):
series_name, series_data = serie
dataset_labels = []
dataset_values = []
for data_point in series_data:
dataset_labels.extend(data_point.keys())
dataset_values.extend(data_point.values())
labels = dataset_labels
dataset = {
'label': series_name,
'data': dataset_values,
}
dataset.update(
ChartJSLine.dataset_palette[
count % len(ChartJSLine.dataset_palette)
]
)
datasets.append(dataset)
data = {
'labels': labels,
'datasets': datasets,
}
return json.dumps(data)