# -*- 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.filtered('customer'): 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', 'child_ids.invoice_ids', 'child_ids.invoice_ids.state', 'child_ids.invoice_ids.amount_total', 'child_ids.invoice_ids.residual', 'child_ids.invoice_ids.company_id.invoice_unpaid_margin') def _compute_risk_invoice(self): def sum_group(group, field): return sum([x[field] for x in group if x['partner_id'][0] in partner_ids]) customers = self.filtered('customer') if not customers: return # pragma: no cover max_date = self._max_risk_date_due() AccountInvoice = self.env['account.invoice'] partners = customers | customers.mapped('child_ids') domain = [('type', 'in', ['out_invoice', 'out_refund']), ('partner_id', 'in', partners.ids)] draft_group = AccountInvoice.read_group( domain + [('state', 'in', ['draft', 'proforma', 'proforma2'])], ['partner_id', 'amount_total'], ['partner_id']) open_group = AccountInvoice.read_group( domain + [('state', '=', 'open'), ('date_due', '>=', max_date)], ['partner_id', 'residual'], ['partner_id']) unpaid_group = AccountInvoice.read_group( domain + [('state', '=', 'open'), '|', ('date_due', '=', False), ('date_due', '<', max_date)], ['partner_id', 'residual'], ['partner_id']) for partner in customers: partner_ids = (partner | partner.child_ids).ids partner.risk_invoice_draft = sum_group(draft_group, 'amount_total') partner.risk_invoice_open = sum_group(open_group, 'residual') partner.risk_invoice_unpaid = sum_group(unpaid_group, 'residual') @api.multi @api.depends('credit', 'risk_invoice_open', 'risk_invoice_unpaid', 'child_ids.credit', 'child_ids.risk_invoice_open', 'child_ids.risk_invoice_unpaid') def _compute_risk_account_amount(self): for partner in self.filtered('customer'): 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.filtered('customer'): 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 partner.credit_limit and 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): res = [] for x in self._risk_field_list(): res.extend((x[0], x[1], x[2], 'child_ids.%s' % x[0], 'child_ids.%s' % x[1], 'child_ids.%s' % x[2])) res.extend(('credit_limit', 'child_ids.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