############################################################################## # # Copyright (c) 2010 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. # ############################################################################## from AccessControl import ClassSecurityInfo from Products.ERP5Type import Permissions from Products.ERP5Legacy.Document.Rule import Rule from Products.ERP5.Document.PredicateMatrix import PredicateMatrix class PaymentSimulationRule(Rule, PredicateMatrix): """ Payment Simulation Rule generates payment simulation movements from accounting / invoice transaction simulation movements. """ # CMF Type Definition meta_type = 'ERP5 Payment Simulation Rule' portal_type = 'Payment Simulation Rule' add_permission = Permissions.AddPortalContent # Declarative security security = ClassSecurityInfo() security.declareObjectProtected(Permissions.AccessContentsInformation) def _generatePrevisionList(self, applied_rule, **kw): """ Generate a list of dictionaries, that contain calculated content of current Simulation Movements in applied rule. based on its context (parent movement, delivery, configuration ...) These previsions are returned as dictionaries. * source and destination (i.e. account) are provided by rule cells. * start_date, stop_date and quantity are calculated according to payment conditions. """ # Find an input movement and using Payment Conditions. # XXX we also need to support local Payment Conditions, that are not # provided by BPM. movement_and_tuple_list = self._getInputMovementAndPathTupleList( applied_rule) input_movement = movement_and_tuple_list[0][0] payment_condition_list = [] # try to find local payment conditions from the upper level delivery rule = applied_rule movement = input_movement delivery = movement.getDeliveryValue() while delivery is None and not(rule.isRootAppliedRule()): rule = movement.getParentValue() movement = rule.getParentValue() delivery = movement.getDeliveryValue() if delivery is not None: payment_condition_list = delivery.getPaymentConditionValueList() # try to find payment conditions in specialised trade conditions if len(payment_condition_list) == 0: specialise = input_movement.getSpecialiseValue() if specialise is None and delivery is not None: specialise = delivery.getSpecialiseValue() if specialise is not None: payment_condition_list = specialise.getPaymentConditionValueList() # try to use payment conditions in BPM configuration if len(payment_condition_list) == 0: payment_condition_list = [x[1] for x in movement_and_tuple_list if x[1] is not None] kw = self._getExpandablePropertyDict(applied_rule, input_movement, None) prevision_list = [] # Find a matching cell cell = self._getMatchingCell(input_movement) if cell is not None : # else, we do nothing for payment_condition in payment_condition_list: # XXX if (payment_condition.getCalculationScript(input_movement) is not None or payment_condition.getEfficiency() != 1): raise NotImplementedError #amount, = payment_condition.getAggregatedAmountList((input_movement,)) #start_date = amount.getStartDate() # does it depend on any property #stop_date = amount.getStopDate() # of payment_condition ? #quantity = amount.getQuantity() start_date = input_movement.getStartDate() stop_date = input_movement.getStopDate() quantity = input_movement.getTotalPrice() * payment_condition.getQuantity(1) payment_mode = payment_condition.getPaymentMode() # one for payable prevision_line = kw.copy() prevision_line.update( start_date=start_date, stop_date=stop_date, source=input_movement.getSource(), destination=input_movement.getDestination(), payment_mode=payment_mode, quantity=-quantity ) prevision_list.append(prevision_line) # one for cash, bank etc. payment_rule_cell_line_list = cell.objectValues() assert len(payment_rule_cell_line_list) == 1 payment_rule_cell_line = payment_rule_cell_line_list[0] prevision_line = kw.copy() prevision_line.update( start_date=start_date, stop_date=stop_date, source=payment_rule_cell_line.getSource(), destination=payment_rule_cell_line.getDestination(), payment_mode=payment_mode, quantity=quantity ) prevision_list.append(prevision_line) return prevision_list security.declareProtected(Permissions.ModifyPortalContent, 'expand') def expand(self, applied_rule, **kw): """Expands the current movement downward. """ return Rule._expand(self, applied_rule, **kw) # Matrix related security.declareProtected( Permissions.ModifyPortalContent, 'newCellContent' ) def newCellContent(self, id, portal_type='Accounting Rule Cell', **kw): """Overriden to specify default portal type """ return self.newContent(id=id, portal_type=portal_type, **kw)