From d91746bb91db7a14f2d09b8ac458526b724123a4 Mon Sep 17 00:00:00 2001 From: ThomasBinsfeld Date: Mon, 29 Feb 2016 12:28:10 +0100 Subject: [PATCH 1/6] [ADD] AccountingNone (singleton) to differentiate balances among which the debit and the credit are zero and balances among which debit and credit nullify --- mis_builder/__openerp__.py | 0 mis_builder/models/accounting_none.py | 136 ++++++++++++++++++++++++++ mis_builder/models/aep.py | 6 +- mis_builder/models/mis_builder.py | 8 +- 4 files changed, 145 insertions(+), 5 deletions(-) mode change 100644 => 100755 mis_builder/__openerp__.py create mode 100755 mis_builder/models/accounting_none.py mode change 100644 => 100755 mis_builder/models/aep.py mode change 100644 => 100755 mis_builder/models/mis_builder.py diff --git a/mis_builder/__openerp__.py b/mis_builder/__openerp__.py old mode 100644 new mode 100755 diff --git a/mis_builder/models/accounting_none.py b/mis_builder/models/accounting_none.py new file mode 100755 index 00000000..bcf15ecb --- /dev/null +++ b/mis_builder/models/accounting_none.py @@ -0,0 +1,136 @@ +# -*- coding: utf-8 -*- + + +""" +Provides the AccountingNone singleton + +AccountingNone is a null value that dissolves in basic arithmetic operations, +as illustrated in the examples below + +>>> 1 + 1 +2 +>>> 1 + AccountingNone +1 +>>> AccountingNone + 1 +1 +>>> AccountingNone + None +AccountingNone +>>> +AccountingNone +AccountingNone +>>> -AccountingNone +AccountingNone +>>> -(AccountingNone) +AccountingNone +>>> AccountingNone - 1 +-1 +>>> 1 - AccountingNone +1 +>>> AccountingNone - None +AccountingNone +>>> AccountingNone / 2 +0.0 +>>> 2 / AccountingNone +Traceback (most recent call last): + ... +ZeroDivisionError +>>> AccountingNone / AccountingNone +AccountingNone +>>> AccountingNone // 2 +0.0 +>>> 2 // AccountingNone +Traceback (most recent call last): + ... +ZeroDivisionError +>>> AccountingNone // AccountingNone +AccountingNone +>>> AccountingNone * 2 +0.0 +>>> 2 * AccountingNone +0.0 +>>> AccountingNone * AccountingNone +AccountingNone +>>> AccountingNone * None +AccountingNone +""" + + +class AccountingNoneType(object): + + def __add__(self, other): + if other is None: + return AccountingNone + return other + + __radd__ = __add__ + + def __sub__(self, other): + if other is None: + return AccountingNone + return -other + + def __rsub__(self, other): + if other is None: + return AccountingNone + return other + + def __iadd__(self, other): + if other is None: + return AccountingNone + return other + + def __isub__(self, other): + if other is None: + return AccountingNone + return -other + + def __pos__(self): + return self + + def __neg__(self): + return self + + def __floordiv__(self, other): + """ + Overload of the // operator + """ + if other is AccountingNone: + return AccountingNone + return 0.0 + + def __rfloordiv__(self, other): + raise ZeroDivisionError + + def __truediv__(self, other): + """ + Overload of the / operator + """ + if other is AccountingNone: + return AccountingNone + return 0.0 + + def __rtruediv__(self, other): + raise ZeroDivisionError + + def __mul__(self, other): + if other is None or other is AccountingNone: + return AccountingNone + return 0.0 + + def __rmul__(self, other): + if other is None or other is AccountingNone: + return AccountingNone + return 0.0 + + def __repr__(self): + return 'AccountingNone' + + def __unicode__(self): + return '' + + +AccountingNone = AccountingNoneType() + + +if __name__ == '__main__': + import doctest + doctest.testmod() diff --git a/mis_builder/models/aep.py b/mis_builder/models/aep.py old mode 100644 new mode 100755 index ffa344ae..8562d863 --- a/mis_builder/models/aep.py +++ b/mis_builder/models/aep.py @@ -9,6 +9,7 @@ from openerp.exceptions import Warning as UserError from openerp.models import expression from openerp.tools.safe_eval import safe_eval from openerp.tools.translate import _ +from .accounting_none import AccountingNone MODE_VARIATION = 'p' MODE_INITIAL = 'i' @@ -348,12 +349,13 @@ class AccountingExpressionProcessor(object): field, mode, account_codes, domain = self._parse_match_object(mo) key = (domain, mode) account_ids_data = self._data[key] - v = 0.0 + v = AccountingNone for account_code in account_codes: account_ids = self._account_ids_by_code[account_code] for account_id in account_ids: debit, credit = \ - account_ids_data.get(account_id, (0.0, 0.0)) + account_ids_data.get(account_id, + (AccountingNone, AccountingNone)) if field == 'bal': v += debit - credit elif field == 'deb': diff --git a/mis_builder/models/mis_builder.py b/mis_builder/models/mis_builder.py old mode 100644 new mode 100755 index 3872233a..c206a00c --- a/mis_builder/models/mis_builder.py +++ b/mis_builder/models/mis_builder.py @@ -16,6 +16,7 @@ from openerp.tools.safe_eval import safe_eval from .aep import AccountingExpressionProcessor as AEP from .aggregate import _sum, _avg, _min, _max +from .accounting_none import AccountingNone _logger = logging.getLogger(__name__) @@ -140,8 +141,8 @@ class MisReportKpi(models.Model): def render(self, lang_id, value): """ render a KPI value as a unicode string, ready for display """ assert len(self) == 1 - if value is None: - return '#N/A' + if value is None or value is AccountingNone: + return '' elif self.type == 'num': return self._render_num(lang_id, value, self.divider, self.dp, self.prefix, self.suffix) @@ -497,6 +498,7 @@ class MisReportInstancePeriod(models.Model): 'max': _max, 'len': len, 'avg': _avg, + 'AccountingNone': AccountingNone, } localdict.update(self._fetch_queries()) @@ -544,7 +546,7 @@ class MisReportInstancePeriod(models.Model): AEP.has_account_var(kpi.expression)) res[kpi.name] = { - 'val': kpi_val, + 'val': None if kpi_val is AccountingNone else kpi_val, 'val_r': kpi_val_rendered, 'val_c': kpi_val_comment, 'style': kpi_style, From 2958d3fa8e9a9ac530dd58f1c8adb97becc7c3c8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phane=20Bidoul?= Date: Sun, 13 Mar 2016 22:17:04 +0100 Subject: [PATCH 2/6] [FIX] reset permissions that should not have changed --- mis_builder/__openerp__.py | 0 mis_builder/models/accounting_none.py | 0 mis_builder/models/aep.py | 0 mis_builder/models/mis_builder.py | 0 4 files changed, 0 insertions(+), 0 deletions(-) mode change 100755 => 100644 mis_builder/__openerp__.py mode change 100755 => 100644 mis_builder/models/accounting_none.py mode change 100755 => 100644 mis_builder/models/aep.py mode change 100755 => 100644 mis_builder/models/mis_builder.py diff --git a/mis_builder/__openerp__.py b/mis_builder/__openerp__.py old mode 100755 new mode 100644 diff --git a/mis_builder/models/accounting_none.py b/mis_builder/models/accounting_none.py old mode 100755 new mode 100644 diff --git a/mis_builder/models/aep.py b/mis_builder/models/aep.py old mode 100755 new mode 100644 diff --git a/mis_builder/models/mis_builder.py b/mis_builder/models/mis_builder.py old mode 100755 new mode 100644 From 9b626fe5daa27c9e02ac8e4683671475f20a5b11 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phane=20Bidoul?= Date: Sun, 13 Mar 2016 22:19:41 +0100 Subject: [PATCH 3/6] [IMP] improve AccountingNone wrt comparisons mainly --- mis_builder/models/accounting_none.py | 67 ++++++++++++++++++++++----- 1 file changed, 55 insertions(+), 12 deletions(-) diff --git a/mis_builder/models/accounting_none.py b/mis_builder/models/accounting_none.py index bcf15ecb..8986ed41 100644 --- a/mis_builder/models/accounting_none.py +++ b/mis_builder/models/accounting_none.py @@ -5,7 +5,8 @@ Provides the AccountingNone singleton AccountingNone is a null value that dissolves in basic arithmetic operations, -as illustrated in the examples below +as illustrated in the examples below. In comparisons, AccountingNone behaves +the same as zero. >>> 1 + 1 2 @@ -51,6 +52,34 @@ AccountingNone AccountingNone >>> AccountingNone * None AccountingNone +>>> None * AccountingNone +AccountingNone +>>> str(AccountingNone) +'' +>>> bool(AccountingNone) +False +>>> AccountingNone > 0 +False +>>> AccountingNone < 0 +False +>>> AccountingNone < 1 +True +>>> AccountingNone > 1 +False +>>> 0 < AccountingNone +False +>>> 0 > AccountingNone +False +>>> 1 < AccountingNone +False +>>> 1 > AccountingNone +True +>>> AccountingNone == 0 +True +>>> AccountingNone == 0.0 +True +>>> AccountingNone == None +True """ @@ -89,10 +118,15 @@ class AccountingNoneType(object): def __neg__(self): return self + def __div__(self, other): + if other is AccountingNone: + return AccountingNone + return 0.0 + + def __rdiv__(self, other): + raise ZeroDivisionError + def __floordiv__(self, other): - """ - Overload of the // operator - """ if other is AccountingNone: return AccountingNone return 0.0 @@ -101,9 +135,6 @@ class AccountingNoneType(object): raise ZeroDivisionError def __truediv__(self, other): - """ - Overload of the / operator - """ if other is AccountingNone: return AccountingNone return 0.0 @@ -116,17 +147,29 @@ class AccountingNoneType(object): return AccountingNone return 0.0 - def __rmul__(self, other): - if other is None or other is AccountingNone: - return AccountingNone - return 0.0 + __rmul__ = __mul__ def __repr__(self): return 'AccountingNone' - def __unicode__(self): + def __str__(self): return '' + def __nonzero__(self): + return False + + def __bool__(self): + return False + + def __eq__(self, other): + return other == 0 or other is None or other is AccountingNone + + def __lt__(self, other): + return 0 < other + + def __gt__(self, other): + return 0 > other + AccountingNone = AccountingNoneType() From fe4fa678fb52c5efc0dceb96df7a85eec28645ef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phane=20Bidoul?= Date: Sun, 13 Mar 2016 22:56:39 +0100 Subject: [PATCH 4/6] Add copyright header and __all__ to accounting_none.py --- mis_builder/models/accounting_none.py | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/mis_builder/models/accounting_none.py b/mis_builder/models/accounting_none.py index 8986ed41..57f7f549 100644 --- a/mis_builder/models/accounting_none.py +++ b/mis_builder/models/accounting_none.py @@ -1,8 +1,9 @@ # -*- coding: utf-8 -*- - - +# © 2016 Thomas Binsfeld +# © 2016 ACSONE SA/NV () +# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). """ -Provides the AccountingNone singleton +Provides the AccountingNone singleton. AccountingNone is a null value that dissolves in basic arithmetic operations, as illustrated in the examples below. In comparisons, AccountingNone behaves @@ -82,6 +83,8 @@ True True """ +__all__ = ['AccountingNone'] + class AccountingNoneType(object): From 2b1f182d62a623540abb976b37d07246c8e3f924 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phane=20Bidoul?= Date: Mon, 14 Mar 2016 12:14:46 +0100 Subject: [PATCH 5/6] [FIX] fix comparison rendering in presence of AccountingNone --- mis_builder/models/accounting_none.py | 5 +++++ mis_builder/models/mis_builder.py | 12 +++++++----- 2 files changed, 12 insertions(+), 5 deletions(-) diff --git a/mis_builder/models/accounting_none.py b/mis_builder/models/accounting_none.py index 57f7f549..5249e540 100644 --- a/mis_builder/models/accounting_none.py +++ b/mis_builder/models/accounting_none.py @@ -27,6 +27,8 @@ AccountingNone -1 >>> 1 - AccountingNone 1 +>>> abs(AccountingNone) +AccountingNone >>> AccountingNone - None AccountingNone >>> AccountingNone / 2 @@ -115,6 +117,9 @@ class AccountingNoneType(object): return AccountingNone return -other + def __abs__(self): + return self + def __pos__(self): return self diff --git a/mis_builder/models/mis_builder.py b/mis_builder/models/mis_builder.py index c206a00c..19f212f7 100644 --- a/mis_builder/models/mis_builder.py +++ b/mis_builder/models/mis_builder.py @@ -156,17 +156,19 @@ class MisReportKpi(models.Model): average_value, average_base_value): """ render the comparison of two KPI values, ready for display """ assert len(self) == 1 - if value is None or base_value is None: - return '' + if value is None: + value = AccountingNone + if base_value is None: + base_value = AccountingNone if self.type == 'pct': return self._render_num( lang_id, value - base_value, 0.01, self.dp, '', _('pp'), sign='+') elif self.type == 'num': - if average_value: + if value and average_value: value = value / float(average_value) - if average_base_value: + if base_value and average_base_value: base_value = base_value / float(average_base_value) if self.compare_method == 'diff': return self._render_num( @@ -174,7 +176,7 @@ class MisReportKpi(models.Model): value - base_value, self.divider, self.dp, self.prefix, self.suffix, sign='+') elif self.compare_method == 'pct': - if round(base_value, self.dp) != 0: + if base_value and round(base_value, self.dp) != 0: return self._render_num( lang_id, (value - base_value) / abs(base_value), From e8cb112c21ea2fda0e531bf0b8539d0c16727ae5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?St=C3=A9phane=20Bidoul?= Date: Fri, 18 Mar 2016 13:09:01 +0100 Subject: [PATCH 6/6] mis_builder: render blank instead of +0 comparison columns --- mis_builder/models/mis_builder.py | 38 ++++++++++++++++++++----------- 1 file changed, 25 insertions(+), 13 deletions(-) diff --git a/mis_builder/models/mis_builder.py b/mis_builder/models/mis_builder.py index 19f212f7..62167b14 100644 --- a/mis_builder/models/mis_builder.py +++ b/mis_builder/models/mis_builder.py @@ -154,33 +154,45 @@ class MisReportKpi(models.Model): def render_comparison(self, lang_id, value, base_value, average_value, average_base_value): - """ render the comparison of two KPI values, ready for display """ + """ render the comparison of two KPI values, ready for display + + If the difference is 0, an empty string is returned. + """ assert len(self) == 1 if value is None: value = AccountingNone if base_value is None: base_value = AccountingNone if self.type == 'pct': - return self._render_num( - lang_id, - value - base_value, - 0.01, self.dp, '', _('pp'), sign='+') + delta = value - base_value + if delta and round(delta, self.dp) != 0: + return self._render_num( + lang_id, + delta, + 0.01, self.dp, '', _('pp'), + sign='+') elif self.type == 'num': if value and average_value: value = value / float(average_value) if base_value and average_base_value: base_value = base_value / float(average_base_value) if self.compare_method == 'diff': - return self._render_num( - lang_id, - value - base_value, - self.divider, self.dp, self.prefix, self.suffix, sign='+') - elif self.compare_method == 'pct': - if base_value and round(base_value, self.dp) != 0: + delta = value - base_value + if delta and round(delta, self.dp) != 0: return self._render_num( lang_id, - (value - base_value) / abs(base_value), - 0.01, self.dp, '', '%', sign='+') + delta, + self.divider, self.dp, self.prefix, self.suffix, + sign='+') + elif self.compare_method == 'pct': + if base_value and round(base_value, self.dp) != 0: + delta = (value - base_value) / abs(base_value) + if delta and round(delta, self.dp) != 0: + return self._render_num( + lang_id, + delta, + 0.01, self.dp, '', '%', + sign='+') return '' def _render_num(self, lang_id, value, divider,