[Erp5-report] r30725 - in /erp5/trunk/products/ERP5: Document/ mixin/

nobody at svn.erp5.org nobody at svn.erp5.org
Wed Nov 18 14:16:54 CET 2009


Author: kazuhiko
Date: Wed Nov 18 14:16:53 2009
New Revision: 30725

URL: http://svn.erp5.org?rev=30725&view=rev
Log:
move some shareable part to mixin/divergence_tester.py.

Added:
    erp5/trunk/products/ERP5/mixin/divergence_tester.py
Modified:
    erp5/trunk/products/ERP5/Document/FloatDivergenceTester.py

Modified: erp5/trunk/products/ERP5/Document/FloatDivergenceTester.py
URL: http://svn.erp5.org/erp5/trunk/products/ERP5/Document/FloatDivergenceTester.py?rev=30725&r1=30724&r2=30725&view=diff
==============================================================================
--- erp5/trunk/products/ERP5/Document/FloatDivergenceTester.py [utf8] (original)
+++ erp5/trunk/products/ERP5/Document/FloatDivergenceTester.py [utf8] Wed Nov 18 14:16:53 2009
@@ -32,10 +32,10 @@
 from AccessControl import ClassSecurityInfo
 
 from Products.ERP5.Document.Predicate import Predicate
-from Products.ERP5Type.DivergenceMessage import DivergenceMessage
 from Products.ERP5Type import Permissions, PropertySheet, interfaces
+from Products.ERP5.mixin.divergence_tester import DivergenceTesterMixin
 
-class FloatDivergenceTester(Predicate):
+class FloatDivergenceTester(Predicate, DivergenceTesterMixin):
   """
   The purpose of this divergence tester is to check the
   consistency between delivery movement and simulation movement
@@ -61,84 +61,9 @@
   # Declarative interfaces
   zope.interface.implements( interfaces.IDivergenceTester, )
 
-  def testDivergence(self, simulation_movement):
-    """
-    Tests if simulation_movement is divergent. Returns False (0)
-    or True (1).
-
-    If decision_movement is a simulation movement, use
-    the recorded properties instead of the native ones.
-
-    simulation_movement -- a simulation movement
-    """
-    return self.explain(simulation_movement) is not None
-
-  def explain(self, simulation_movement):
-    """
-    Returns a single message which explain the nature of
-    the divergence of simulation_movement with its related
-    delivery movement.
-
-    If decision_movement is a simulation movement, use
-    the recorded properties instead of the native ones.
-
-    simulation_movement -- a simulation movement
-
-    NOTE: this approach is incompatible with previous
-    API which was returning a list.
-
-    NOTE: should we provide compatibility here ?
-    """
-    delivery_movement = simulation_movement.getDeliveryValue()
-    compare_result = self._compare(simulation_movement, delivery_movement)
-    if compare_result is None:
-      return None
-    else:
-      prevision_value, decision_value, message, mapping = compare_result
-      return DivergenceMessage(
-        object_relative_url=delivery_movement.getRelativeUrl(),
-        simulation_movement=simulation_movement,
-        decision_value=decision_value,
-        prevision_value=prevision_value,
-        tested_property=self.getTestedProperty(),
-        message=message,
-        mapping=mapping
-        )
-
-  def generateHashKey(self, movement):
-    """
-    Returns a hash key which can be used to optimise the
-    matching algorithm between movements. The purpose
-    of this hash key is to reduce the size of lists of
-    movements which need to be compared using the compare
-    method (quadratic complexity).
-
-    If decision_movement is a simulation movement, use
-    the recorded properties instead of the native ones.
-    """
-    return '%s/%s' % (self.getPortalType(), self.getTestedProperty())
-
-  def compare(self, prevision_movement, decision_movement):
-    """
-    Returns True if simulation_movement and delivery_movement
-    match. Returns False else. The method is asymmetric and
-    the order of parameter matters. For example, a sourcing
-    rule may use a tester which makes sure that movements are
-    delivered no sooner than 2 weeks before production but
-    no later than the production date.
-
-    If decision_movement is a simulation movement, use
-    the recorded properties instead of the native ones.
-
-    prevision_movement -- a simulation movement (prevision)
-
-    decision_movement -- a delivery movement (decision)
-    """
-    return (self._compare(prevision_movement, decision_movement) is None)
-
   def _compare(self, prevision_movement, decision_movement):
     """
-    If prevision_movement and decision_movement dont match, it returns a
+    If prevision_movement and decision_movement don't match, it returns a
     list : (prevision_value, decision_value, message, mapping)
     """
     tested_property = self.getTestedProperty()
@@ -222,32 +147,23 @@
     # * decimal_rounding_option
     # * decimal_exponent
 
-  def update(self, prevision_movement, decision_movement):
+  def getUpdatablePropertyDict(self, prevision_movement, decision_movement):
     """
-    Updates decision_movement with properties from
-    prevision_movement so that next call to
-    compare returns True. This method is normally
-    invoked to copy properties from simulation movements
-    to delivery movements. It is also invoked to copy
-    properties from temp simulation movements of
-    Aggregated Amount Lists to pre-existing simulation
-    movements.
-
-    If decision_movement is a simulation movement, then
-    do not update recorded properties.
+    Returns a list of properties to update on decision_movement
+    prevision_movement so that next call to compare returns True.
 
     prevision_movement -- a simulation movement (prevision)
 
     decision_movement -- a delivery movement (decision)
-
-    NOTE: recorded (forced) properties are not updated by
-    expand.
-
-    NOTE2: it is still unknown how to update properties from
-    a simulation movement to the relevant level of
-    delivery / line / cell.
     """
-    raise NotImplementedError
+    tested_property = self.getTestedProperty()
+    if prevision_movement.isPropertyRecorded(tested_property):
+      prevision_value = prevision_movement.getRecordedProperty(tested_property)
+      if isinstance(prevision_value, (list, tuple)):
+        prevision_value = prevision_value[0]
+    else:
+      prevision_value = prevision_movement.getProperty(tested_property)
+    return {tested_property:prevision_value}
 
   def accept(self, simulation_movement):
     """

Added: erp5/trunk/products/ERP5/mixin/divergence_tester.py
URL: http://svn.erp5.org/erp5/trunk/products/ERP5/mixin/divergence_tester.py?rev=30725&view=auto
==============================================================================
--- erp5/trunk/products/ERP5/mixin/divergence_tester.py (added)
+++ erp5/trunk/products/ERP5/mixin/divergence_tester.py [utf8] Wed Nov 18 14:16:53 2009
@@ -1,0 +1,189 @@
+# -*- coding: utf-8 -*-
+##############################################################################
+#
+# Copyright (c) 2009 Nexedi SA and Contributors. All Rights Reserved.
+#
+# WARNING: This program as such is intended to be used by professional
+# programmers who take the whole responsibility of assessing all potential
+# consequences resulting from its eventual inadequacies and bugs
+# End users who are looking for a ready-to-use solution with commercial
+# guarantees and support are strongly adviced to contract a Free Software
+# Service Company
+#
+# This program is Free Software; you can redistribute it and/or
+# modify it under the terms of the GNU General Public License
+# as published by the Free Software Foundation; either version 2
+# of the License, or (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+#
+##############################################################################
+
+import zope.interface
+from AccessControl import ClassSecurityInfo
+from Products.ERP5Type import Permissions, interfaces
+from Products.ERP5Type.DivergenceMessage import DivergenceMessage
+
+class DivergenceTesterMixin:
+  """
+  Provides generic methods and helper methods to implement
+  IDivergenceTester
+  """
+  # Declarative security
+  security = ClassSecurityInfo()
+  security.declareObjectProtected(Permissions.AccessContentsInformation)
+
+  # Declarative interfaces
+  zope.interface.implements(interfaces.IDivergenceTester,)
+
+  # Implementation of IDivergenceTester
+  def testDivergence(self, simulation_movement):
+    """
+    Tests if simulation_movement is divergent. Returns False (0)
+    or True (1).
+
+    If decision_movement is a simulation movement, use
+    the recorded properties instead of the native ones.
+
+    simulation_movement -- a simulation movement
+    """
+    return self.explain(simulation_movement) is not None
+
+  def explain(self, simulation_movement):
+    """
+    Returns a single message which explain the nature of
+    the divergence of simulation_movement with its related
+    delivery movement.
+
+    If decision_movement is a simulation movement, use
+    the recorded properties instead of the native ones.
+
+    simulation_movement -- a simulation movement
+
+    NOTE: this approach is incompatible with previous
+    API which was returning a list.
+
+    NOTE: should we provide compatibility here ?
+    """
+    delivery_movement = simulation_movement.getDeliveryValue()
+    compare_result = self._compare(simulation_movement, delivery_movement)
+    if compare_result is None:
+      return None
+    else:
+      prevision_value, decision_value, message, mapping = compare_result
+      return DivergenceMessage(
+        object_relative_url=delivery_movement.getRelativeUrl(),
+        simulation_movement=simulation_movement,
+        decision_value=decision_value,
+        prevision_value=prevision_value,
+        tested_property=self.getTestedProperty(),
+        message=message,
+        mapping=mapping
+        )
+
+  def generateHashKey(self, movement):
+    """
+    Returns a hash key which can be used to optimise the
+    matching algorithm between movements. The purpose
+    of this hash key is to reduce the size of lists of
+    movements which need to be compared using the compare
+    method (quadratic complexity).
+
+    If decision_movement is a simulation movement, use
+    the recorded properties instead of the native ones.
+    """
+    return '%s/%s' % (self.getPortalType(), self.getTestedProperty())
+
+  def compare(self, prevision_movement, decision_movement):
+    """
+    Returns True if simulation_movement and delivery_movement
+    match. Returns False else. The method is asymmetric and
+    the order of parameter matters. For example, a sourcing
+    rule may use a tester which makes sure that movements are
+    delivered no sooner than 2 weeks before production but
+    no later than the production date.
+
+    If decision_movement is a simulation movement, use
+    the recorded properties instead of the native ones.
+
+    prevision_movement -- a simulation movement (prevision)
+
+    decision_movement -- a delivery movement (decision)
+    """
+    return (self._compare(prevision_movement, decision_movement) is None)
+
+  def update(self, prevision_movement, decision_movement):
+    """
+    Updates decision_movement with properties from
+    prevision_movement so that next call to
+    compare returns True. This method is normally
+    invoked to copy properties from simulation movements
+    to delivery movements. It is also invoked to copy
+    properties from temp simulation movements of
+    Aggregated Amount Lists to pre-existing simulation
+    movements.
+
+    If decision_movement is a simulation movement, then
+    do not update recorded properties.
+
+    prevision_movement -- a simulation movement (prevision)
+
+    decision_movement -- a delivery movement (decision)
+
+    NOTE: recorded (forced) properties are not updated by
+    expand.
+
+    NOTE2: it is still unknown how to update properties from
+    a simulation movement to the relevant level of
+    delivery / line / cell.
+    """
+    decision_movement.edit(
+      **self.getUpdatablePropertyDict(prevision_movement, decision_movement))
+
+  # Placeholder for methods to override
+  def _compare(self, prevision_movement, decision_movement):
+    """
+    If prevision_movement and decision_movement don't match, it returns a
+    list : (prevision_value, decision_value, message, mapping)
+    """
+    raise NotImplementedError
+
+  def getUpdatablePropertyDict(prevision_movement, decision_movement):
+    """
+    Returns a list of properties to update on decision_movement
+    prevision_movement so that next call to compare returns True.
+
+    prevision_movement -- a simulation movement (prevision)
+
+    decision_movement -- a delivery movement (decision)
+    """
+    raise NotImplementedError
+
+  def accept(self, simulation_movement):
+    """
+    Copies the properties handled by the divergence tester
+    from the related delivery movement to simulation_movement.
+
+    NOTE: the future existence of this method is still unknown
+    because it is likely to be implemented in TargetSolver
+    instead.
+    """
+    raise NotImplementedError
+
+  def adopt(self, simulation_movement):
+    """
+    Copies the properties handled by the divergence tester
+    from simulation_movement to the related delivery movement
+
+    NOTE: the future existence of this method is still unknown
+    because it is likely to be implemented in TargetSolver
+    instead.
+    """
+    raise NotImplementedError




More information about the Erp5-report mailing list