# -*- coding: utf-8 -*- # © 2016 Carlos Dauden # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). from datetime import datetime from dateutil.relativedelta import relativedelta from openerp import api, fields, models class ResPartner(models.Model): _inherit = 'res.partner' risk_invoice_draft_include = fields.Boolean( string='Include Draft Invoices', help='Full risk computation') risk_invoice_draft_limit = fields.Monetary( string='Limit In Draft Invoices', help='Set 0 if it is not locked') risk_invoice_draft = fields.Monetary( compute='_compute_risk_invoice', store=True, string='Total Draft Invoices', help='Total amount of invoices in Draft or Pro-forma state') risk_invoice_open_include = fields.Boolean( string='Include Open Invoices', help='Full risk computation') risk_invoice_open_limit = fields.Monetary( string='Limit In Open Invoices', help='Set 0 if it is not locked') risk_invoice_open = fields.Monetary( compute='_compute_risk_invoice', store=True, string='Total Open Invoices', help='Residual amount of invoices in Open state and the date due is ' 'not exceeded, considering Due Margin set in account ' 'settings') risk_invoice_unpaid_include = fields.Boolean( string='Include Unpaid Invoices', help='Full risk computation') risk_invoice_unpaid_limit = fields.Monetary( string='Limit In Unpaid Invoices', help='Set 0 if it is not locked') risk_invoice_unpaid = fields.Monetary( compute='_compute_risk_invoice', store=True, string='Total Unpaid Invoices', help='Residual amount of invoices in Open state and the date due is ' 'exceeded, considering Unpaid Margin set in account settings') risk_account_amount_include = fields.Boolean( string='Include Other Account Amount', help='Full risk computation') risk_account_amount_limit = fields.Monetary( string='Limit Other Account Amount', help='Set 0 if it is not locked') risk_account_amount = fields.Monetary( compute='_compute_risk_account_amount', string='Other Account Amount', help='Difference between accounting credit and rest of totals') risk_total = fields.Monetary( compute='_compute_risk_exception', string='Total Risk', help='Sum of total risk included') risk_exception = fields.Boolean( compute='_compute_risk_exception', string='Risk Exception', help='It Indicate if partner risk exceeded') credit_policy = fields.Char() risk_allow_edit = fields.Boolean(compute='_compute_risk_allow_edit') @api.multi def _compute_risk_allow_edit(self): is_editable = self.env.user.has_group( 'base.group_sale_manager') or self.env.user.has_group( 'account.group_account_manager') for partner in self: partner.risk_allow_edit = is_editable @api.multi @api.depends('invoice_ids', 'invoice_ids.state', 'invoice_ids.amount_total', 'invoice_ids.residual', 'invoice_ids.company_id.invoice_unpaid_margin') def _compute_risk_invoice(self): max_date = self._max_risk_date_due() for partner in self: invoices_out = partner.invoice_ids.filtered( lambda x: x.type in ['out_invoice', 'out_refund']) invoices = invoices_out.filtered( lambda x: x.state in ['draft', 'proforma', 'proforma2']) partner.risk_invoice_draft = sum(invoices.mapped('amount_total')) invoices = invoices_out.filtered( lambda x: x.state == 'open' and x.date_due >= max_date) partner.risk_invoice_open = sum(invoices.mapped('residual')) invoices = invoices_out.filtered( lambda x: x.state == 'open' and x.date_due < max_date) partner.risk_invoice_unpaid = sum(invoices.mapped('residual')) @api.multi @api.depends('credit', 'risk_invoice_open', 'risk_invoice_unpaid') def _compute_risk_account_amount(self): for partner in self: partner.risk_account_amount = ( partner.credit - partner.risk_invoice_open - partner.risk_invoice_unpaid) @api.multi @api.depends(lambda x: x._get_depends_compute_risk_exception()) def _compute_risk_exception(self): risk_field_list = self._risk_field_list() for partner in self: amount = 0.0 for risk_field in risk_field_list: field_value = getattr(partner, risk_field[0], 0.0) max_value = getattr(partner, risk_field[1], 0.0) if max_value and field_value > max_value: partner.risk_exception = True if getattr(partner, risk_field[2], False): amount += field_value partner.risk_total = amount if amount > partner.credit_limit: partner.risk_exception = True @api.model def _max_risk_date_due(self): return fields.Date.to_string(datetime.today().date() - relativedelta( days=self.env.user.company_id.invoice_unpaid_margin)) @api.model def _risk_field_list(self): return [ ('risk_invoice_draft', 'risk_invoice_draft_limit', 'risk_invoice_draft_include'), ('risk_invoice_open', 'risk_invoice_open_limit', 'risk_invoice_open_include'), ('risk_invoice_unpaid', 'risk_invoice_unpaid_limit', 'risk_invoice_unpaid_include'), ('risk_account_amount', 'risk_account_amount_limit', 'risk_account_amount_include'), ] @api.model def _get_depends_compute_risk_exception(self): # TODO: Improve code without performance loss tuple_list = self._risk_field_list() res = [x[0] for x in tuple_list] res.extend([x[1] for x in tuple_list]) res.extend([x[2] for x in tuple_list]) res.append('credit_limit') return res @api.model def process_unpaid_invoices(self): today = fields.Date.today() ConfigParameter = self.env['ir.config_parameter'] last_check = ConfigParameter.get_param( 'partner_financial_risk.last_check', default='2016-01-01') invoices = self.env['account.invoice'].search([ ('date_due', '>=', last_check), ('date_due', '<', today)]) invoices.mapped('partner_id')._compute_risk_invoice() ConfigParameter.set_param('partner_financial_risk.last_check', today) return True