Files
account-financial-tools/account_credit_control/wizard/credit_control_communication.py
Guewen Baconnier @ Camptocamp 448cde4807 [FIX] rename cursor to cr to be consistent with the openerp codebase
(lp:c2c-addons/6.1  rev 89.1.41)
2012-10-29 14:47:31 +01:00

190 lines
8.9 KiB
Python

# -*- coding: utf-8 -*-
##############################################################################
#
# Author: Nicolas Bessi
# Copyright 2012 Camptocamp SA
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU Affero General Public License as
# published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
#
##############################################################################
from openerp.osv.orm import TransientModel, fields
from openerp.osv.osv import except_osv
from openerp.tools.translate import _
import netsvc
import logging
logger = logging.getLogger('credit.control.line mailing')
class CreditCommunication(TransientModel):
"""Shell calss used to provide a base model to email template and reporting.
Il use this approche in version 7 a browse record will exist even if not saved"""
_name = "credit.control.communication"
_description = "credit control communication"
_rec_name = 'partner_id'
_columns = {'partner_id': fields.many2one('res.partner', 'Partner', required=True),
'current_policy_level': fields.many2one('credit.control.policy.level',
'Level', required=True),
'credit_control_line_ids': fields.many2many('credit.control.line',
rel='comm_credit_rel',
string='Credit Lines'),
'company_id': fields.many2one('res.company', 'Company',
required=True),
'user_id': fields.many2one('res.users', 'User')}
_defaults = {'company_id': lambda s, cr, uid, c: s.pool.get('res.company')._company_default_get(
cr, uid, 'credit.control.policy', context=c),
'user_id': lambda s, cr, uid, c: uid}
def get_address(self, cr, uid, com_id, context=None):
"""Return a valid address for customer"""
assert not (isinstance(com_id, list) and len(com_id) > 1), \
"com_id: only one id expected"
if isinstance(com_id, list):
com_id = com_id[0]
form = self.browse(cr, uid, com_id, context=context)
part_obj = self.pool.get('res.partner')
adds = part_obj.address_get(cr, uid, [form.partner_id.id],
adr_pref=['invoice', 'default'])
add = adds.get('invoice', adds.get('default'))
if not add:
raise except_osv(_('No address for partner %s') % (form.partner_id.name),
_('Please set one'))
add_obj = self.pool.get('res.partner.address')
return add_obj.browse(cr, uid, add, context=context)
def get_mail(self, cr, uid, com_id, context=None):
"""Return a valid email for customer"""
context = context or {}
if isinstance(com_id, list):
com_id = com_id[0]
form = self.browse(cr, uid, com_id, context=context)
email = form.get_address().email
if not email:
raise except_osv(_('No invoicing or default email for partner %s') %
(form.partner_id.name),
_('Please set one'))
return email
def _get_credit_lines(self, cr, uid, line_ids, partner_id, level_id, context=None):
"""Return credit lines related to a partner and a policy level"""
cr_line_obj = self.pool.get('credit.control.line')
cr_l_ids = cr_line_obj.search(cr,
uid,
[('id', 'in', line_ids),
('partner_id', '=', partner_id),
('policy_level_id', '=', level_id)],
context=context)
#return cr_line_obj.browse(cr, uid, cr_l_ids, context=context)
return cr_l_ids
def _generate_comm_from_credit_line_ids(self, cr, uid, line_ids, context=None):
if not line_ids:
return []
comms = []
sql = ("SELECT distinct partner_id, policy_level_id, credit_control_policy_level.level"
" FROM credit_control_line JOIN credit_control_policy_level "
" ON (credit_control_line.policy_level_id = credit_control_policy_level.id)"
" WHERE credit_control_line.id in %s"
" ORDER by credit_control_policy_level.level")
cr.execute(sql, (tuple(line_ids),))
res = cr.dictfetchall()
for level_assoc in res:
data = {}
data['credit_control_line_ids'] = \
[(6, 0, self._get_credit_lines(cr, uid, line_ids,
level_assoc['partner_id'],
level_assoc['policy_level_id'],
context=context))]
data['partner_id'] = level_assoc['partner_id']
data['current_policy_level'] = level_assoc['policy_level_id']
comm_id = self.create(cr, uid, data, context=context)
comms.append(self.browse(cr, uid, comm_id, context=context))
return comms
def _generate_mails(self, cr, uid, comms, context=None):
"""Generate mail message using template related to level"""
cr_line_obj = self.pool.get('credit.control.line')
mail_temp_obj = self.pool.get('email.template')
mail_message_obj = self.pool.get('mail.message')
mail_ids = []
for comm in comms:
# we want to use a local cr in order to send the maximum
# of email
try:
template = comm.current_policy_level.mail_template_id.id
# FIXME: usage of this TransientModel is a bit weird with
# the generation of the email, because the mail is linked
# with this model, which is transient (only the owner of the
# record can read it, and it is not persistent)
# the template should be linked with the credit control line.
mvalues = mail_temp_obj.generate_email(cr, uid,
template,
comm.id,
context=context)
essential_values = ['subject', 'body_html',
'email_from', 'email_to']
# FIXME: should we really raise an exception?
# the mail object should handle itself theses cases
for val in essential_values:
if not mvalues.get(val):
raise Exception('Mail generation error with %s', val)
mail_id = mail_message_obj.create(cr, uid, mvalues, context=context)
cl_ids = [cl.id for cl in comm.credit_control_line_ids]
# we do not use local cusros else we have a lock
# FIXME: so use a savepoint or find the cause of the lock
# we are not supposed to rollback or commit the main transaction
cr_line_obj.write(cr, uid, cl_ids,
{'mail_message_id': mail_id,
'state': 'sent'}, context=context)
mail_ids.append(mail_id)
except Exception, exc:
logger.error(exc)
cr.rollback()
cl_ids = [cl.id for cl in comm.credit_control_line_ids]
# we do not use local cusros else we have a lock
cr_line_obj.write(cr, uid, cl_ids,
{'state': 'mail_error'}, context=context)
finally:
cr.commit()
return mail_ids
def _generate_report(self, cr, uid, comms, context=None):
"""Will generate a report by inserting mako template of related policy template"""
service = netsvc.LocalService('report.credit_control_summary')
ids = [x.id for x in comms]
result, format = service.create(cr, uid, ids, {}, {})
return result
def _mark_credit_line_as_sent(self, cr, uid, comms, context=None):
line_ids = []
for comm in comms:
line_ids += [x.id for x in comm.credit_control_line_ids]
l_obj = self.pool.get('credit.control.line')
l_obj.write(cr, uid, line_ids, {'state': 'sent'}, context=context)
return line_ids