2010-04-29 13:49:19 +00:00
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
##############################################################################
|
|
|
|
|
#
|
|
|
|
|
# OpenERP, Open Source Management Solution
|
|
|
|
|
# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>)
|
|
|
|
|
#
|
|
|
|
|
# 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 osv import osv, fields
|
2010-05-03 12:30:48 +00:00
|
|
|
|
import time
|
2010-06-14 08:47:02 +00:00
|
|
|
|
import tools
|
2010-06-24 12:57:47 +00:00
|
|
|
|
import binascii
|
|
|
|
|
import email
|
|
|
|
|
from email.header import decode_header
|
2011-01-17 07:45:34 +00:00
|
|
|
|
from email.utils import parsedate
|
2010-06-11 15:53:51 +00:00
|
|
|
|
import base64
|
2010-06-24 12:57:47 +00:00
|
|
|
|
import re
|
2010-06-24 19:53:32 +00:00
|
|
|
|
from tools.translate import _
|
2010-06-25 12:18:47 +00:00
|
|
|
|
import logging
|
2010-07-02 13:44:45 +00:00
|
|
|
|
import xmlrpclib
|
2010-06-25 12:18:47 +00:00
|
|
|
|
|
|
|
|
|
_logger = logging.getLogger('mailgate')
|
2010-04-29 13:49:19 +00:00
|
|
|
|
|
|
|
|
|
class mailgate_thread(osv.osv):
|
|
|
|
|
'''
|
|
|
|
|
Mailgateway Thread
|
|
|
|
|
'''
|
|
|
|
|
_name = 'mailgate.thread'
|
|
|
|
|
_description = 'Mailgateway Thread'
|
2010-06-24 12:57:47 +00:00
|
|
|
|
|
2010-04-29 13:49:19 +00:00
|
|
|
|
_columns = {
|
2010-07-16 10:41:21 +00:00
|
|
|
|
'message_ids': fields.one2many('mailgate.message', 'res_id', 'Messages', readonly=True),
|
2010-06-24 19:53:32 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-08-12 13:25:34 +00:00
|
|
|
|
def copy(self, cr, uid, id, default=None, context=None):
|
|
|
|
|
"""
|
|
|
|
|
Overrides orm copy method.
|
|
|
|
|
@param self: the object pointer
|
|
|
|
|
@param cr: the current row, from the database cursor,
|
|
|
|
|
@param uid: the current user’s ID for security checks,
|
|
|
|
|
@param id: Id of mailgate thread
|
|
|
|
|
@param default: Dictionary of default values for copy.
|
|
|
|
|
@param context: A standard dictionary for contextual values
|
|
|
|
|
"""
|
|
|
|
|
if default is None:
|
|
|
|
|
default = {}
|
|
|
|
|
|
|
|
|
|
default.update({
|
2010-08-13 12:20:05 +00:00
|
|
|
|
'message_ids': [],
|
|
|
|
|
'date_closed': False,
|
|
|
|
|
'date_open': False
|
|
|
|
|
})
|
2010-08-12 13:25:34 +00:00
|
|
|
|
return super(mailgate_thread, self).copy(cr, uid, id, default, context=context)
|
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
def message_new(self, cr, uid, msg, context):
|
|
|
|
|
raise Exception, _('Method is not implemented')
|
|
|
|
|
|
2010-11-19 13:48:01 +00:00
|
|
|
|
def message_update(self, cr, uid, ids, vals={}, msg="", default_act='pending', context=None):
|
2010-06-25 12:18:47 +00:00
|
|
|
|
raise Exception, _('Method is not implemented')
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-06-25 22:02:47 +00:00
|
|
|
|
def message_followers(self, cr, uid, ids, context=None):
|
|
|
|
|
""" Get a list of emails of the people following this thread
|
|
|
|
|
"""
|
|
|
|
|
res = {}
|
|
|
|
|
if isinstance(ids, (str, int, long)):
|
|
|
|
|
ids = [long(ids)]
|
|
|
|
|
for thread in self.browse(cr, uid, ids, context=context):
|
|
|
|
|
l=[]
|
|
|
|
|
for message in thread.message_ids:
|
2011-06-16 13:31:12 +00:00
|
|
|
|
l.append((message.user_id and message.user_id.user_email) or '')
|
2010-06-25 22:02:47 +00:00
|
|
|
|
l.append(message.email_from or '')
|
|
|
|
|
l.append(message.email_cc or '')
|
|
|
|
|
res[thread.id] = l
|
|
|
|
|
return res
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-06-25 12:18:47 +00:00
|
|
|
|
def msg_send(self, cr, uid, id, *args, **argv):
|
2010-06-24 19:53:32 +00:00
|
|
|
|
raise Exception, _('Method is not implemented')
|
|
|
|
|
|
2010-07-02 16:31:01 +00:00
|
|
|
|
def history(self, cr, uid, cases, keyword, history=False, subject=None, email=False, details=None, \
|
|
|
|
|
email_from=False, message_id=False, references=None, attach=None, email_cc=None, \
|
|
|
|
|
email_bcc=None, email_date=None, context=None):
|
2010-05-12 10:12:56 +00:00
|
|
|
|
"""
|
|
|
|
|
@param self: The object pointer
|
|
|
|
|
@param cr: the current row, from the database cursor,
|
|
|
|
|
@param uid: the current user’s ID for security checks,
|
|
|
|
|
@param cases: a browse record list
|
2011-02-08 13:36:40 +00:00
|
|
|
|
@param keyword: Subject of the history item
|
2010-05-12 10:12:56 +00:00
|
|
|
|
@param history: Value True/False, If True it makes entry in case History otherwise in Case Log
|
2010-07-02 16:31:01 +00:00
|
|
|
|
@param email: Email-To / Recipient address
|
|
|
|
|
@param email_from: Email From / Sender address if any
|
|
|
|
|
@param email_cc: Comma-Separated list of Carbon Copy Emails To addresse if any
|
|
|
|
|
@param email_bcc: Comma-Separated list of Blind Carbon Copy Emails To addresses if any
|
|
|
|
|
@param email_date: Email Date string if different from now, in server Timezone
|
2010-08-12 13:25:34 +00:00
|
|
|
|
@param details: Description, Details of case history if any
|
2010-07-02 05:38:19 +00:00
|
|
|
|
@param atach: Attachment sent in email
|
2010-05-12 10:12:56 +00:00
|
|
|
|
@param context: A standard dictionary for contextual values"""
|
2010-05-31 12:46:48 +00:00
|
|
|
|
if context is None:
|
2010-05-12 10:12:56 +00:00
|
|
|
|
context = {}
|
2010-07-02 05:38:19 +00:00
|
|
|
|
if attach is None:
|
|
|
|
|
attach = []
|
|
|
|
|
|
2011-01-17 07:45:34 +00:00
|
|
|
|
if email_date:
|
|
|
|
|
edate = parsedate(email_date)
|
|
|
|
|
if edate is not None:
|
|
|
|
|
email_date = time.strftime('%Y-%m-%d %H:%M:%S', edate)
|
|
|
|
|
|
2010-05-12 10:12:56 +00:00
|
|
|
|
# The mailgate sends the ids of the cases and not the object list
|
2010-07-02 05:38:19 +00:00
|
|
|
|
|
2010-06-11 13:26:24 +00:00
|
|
|
|
if all(isinstance(case_id, (int, long)) for case_id in cases):
|
|
|
|
|
cases = self.browse(cr, uid, cases, context=context)
|
2010-05-12 10:12:56 +00:00
|
|
|
|
|
2010-05-25 06:35:30 +00:00
|
|
|
|
att_obj = self.pool.get('ir.attachment')
|
2010-05-12 10:12:56 +00:00
|
|
|
|
obj = self.pool.get('mailgate.message')
|
2010-05-25 06:35:30 +00:00
|
|
|
|
|
2010-05-12 10:12:56 +00:00
|
|
|
|
for case in cases:
|
2010-09-07 13:21:40 +00:00
|
|
|
|
attachments = []
|
|
|
|
|
for att in attach:
|
2011-03-10 10:39:51 +00:00
|
|
|
|
att_ids = att_obj.search(cr, uid, [('name','=',att[0]), ('res_id', '=', case.id)])
|
2011-03-09 12:30:33 +00:00
|
|
|
|
if att_ids:
|
|
|
|
|
attachments.append(att_ids[0])
|
|
|
|
|
else:
|
2011-01-31 05:34:47 +00:00
|
|
|
|
attachments.append(att_obj.create(cr, uid, {'res_model':case._name,'res_id':case.id, 'name': att[0], 'datas': base64.encodestring(att[1])}))
|
2010-09-07 13:21:40 +00:00
|
|
|
|
|
2010-09-08 12:49:25 +00:00
|
|
|
|
partner_id = hasattr(case, 'partner_id') and (case.partner_id and case.partner_id.id or False) or False
|
|
|
|
|
if not partner_id and case._name == 'res.partner':
|
|
|
|
|
partner_id = case.id
|
2010-05-12 10:12:56 +00:00
|
|
|
|
data = {
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'name': keyword,
|
|
|
|
|
'user_id': uid,
|
|
|
|
|
'model' : case._name,
|
2010-09-08 12:49:25 +00:00
|
|
|
|
'partner_id': partner_id,
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'res_id': case.id,
|
|
|
|
|
'date': time.strftime('%Y-%m-%d %H:%M:%S'),
|
2010-09-14 10:19:00 +00:00
|
|
|
|
'message_id': message_id,
|
2011-02-08 13:36:40 +00:00
|
|
|
|
'description': details,
|
2010-09-07 13:21:40 +00:00
|
|
|
|
'attachment_ids': [(6, 0, attachments)]
|
2010-05-12 10:12:56 +00:00
|
|
|
|
}
|
2010-08-13 12:20:05 +00:00
|
|
|
|
|
2010-09-07 13:21:40 +00:00
|
|
|
|
if history:
|
2010-07-02 16:31:01 +00:00
|
|
|
|
for param in (email, email_cc, email_bcc):
|
|
|
|
|
if isinstance(param, list):
|
|
|
|
|
param = ", ".join(param)
|
2010-06-25 12:18:47 +00:00
|
|
|
|
|
2010-05-24 13:35:02 +00:00
|
|
|
|
data = {
|
2010-08-10 11:44:27 +00:00
|
|
|
|
'name': subject or _('History'),
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'history': True,
|
|
|
|
|
'user_id': uid,
|
|
|
|
|
'model' : case._name,
|
2010-06-11 13:26:24 +00:00
|
|
|
|
'res_id': case.id,
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'date': email_date or time.strftime('%Y-%m-%d %H:%M:%S'),
|
|
|
|
|
'description': details or (hasattr(case, 'description') and case.description or False),
|
|
|
|
|
'email_to': email,
|
2010-07-02 05:38:19 +00:00
|
|
|
|
'email_from': email_from or \
|
2010-06-11 13:26:24 +00:00
|
|
|
|
(hasattr(case, 'user_id') and case.user_id and case.user_id.address_id and \
|
2010-07-02 16:31:01 +00:00
|
|
|
|
case.user_id.address_id.email),
|
|
|
|
|
'email_cc': email_cc,
|
|
|
|
|
'email_bcc': email_bcc,
|
2010-09-08 12:49:25 +00:00
|
|
|
|
'partner_id': partner_id,
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'references': references,
|
|
|
|
|
'message_id': message_id,
|
2010-06-11 13:26:24 +00:00
|
|
|
|
'attachment_ids': [(6, 0, attachments)]
|
|
|
|
|
}
|
2011-03-10 10:39:51 +00:00
|
|
|
|
|
2010-07-01 10:46:31 +00:00
|
|
|
|
obj.create(cr, uid, data, context=context)
|
2010-05-12 10:12:56 +00:00
|
|
|
|
return True
|
2010-04-29 13:49:19 +00:00
|
|
|
|
mailgate_thread()
|
|
|
|
|
|
2010-07-27 15:26:37 +00:00
|
|
|
|
def format_date_tz(date, tz=None):
|
2010-08-13 12:20:05 +00:00
|
|
|
|
if not date:
|
2010-07-27 15:26:37 +00:00
|
|
|
|
return 'n/a'
|
|
|
|
|
format = tools.DEFAULT_SERVER_DATETIME_FORMAT
|
|
|
|
|
return tools.server_to_local_timestamp(date, format, format, tz)
|
|
|
|
|
|
2010-04-29 13:49:19 +00:00
|
|
|
|
class mailgate_message(osv.osv):
|
|
|
|
|
'''
|
|
|
|
|
Mailgateway Message
|
|
|
|
|
'''
|
2010-11-19 13:48:01 +00:00
|
|
|
|
def open_document(self, cr, uid, ids, context=None):
|
2010-10-01 07:54:12 +00:00
|
|
|
|
""" To Open Document
|
|
|
|
|
@param self: The object pointer.
|
|
|
|
|
@param cr: A database cursor
|
|
|
|
|
@param uid: ID of the user currently logged in
|
|
|
|
|
@param ids: the ID of messages
|
|
|
|
|
@param context: A standard dictionary
|
|
|
|
|
"""
|
|
|
|
|
action_data = False
|
|
|
|
|
if ids:
|
|
|
|
|
message_id = ids[0]
|
2010-11-19 13:48:01 +00:00
|
|
|
|
mailgate_data = self.browse(cr, uid, message_id, context=context)
|
2010-10-01 07:54:12 +00:00
|
|
|
|
model = mailgate_data.model
|
|
|
|
|
res_id = mailgate_data.res_id
|
|
|
|
|
|
|
|
|
|
action_pool = self.pool.get('ir.actions.act_window')
|
|
|
|
|
action_ids = action_pool.search(cr, uid, [('res_model', '=', model)])
|
|
|
|
|
if action_ids:
|
|
|
|
|
action_data = action_pool.read(cr, uid, action_ids[0], context=context)
|
|
|
|
|
action_data.update({
|
2010-10-04 05:18:23 +00:00
|
|
|
|
'domain' : "[('id','=',%d)]"%(res_id),
|
2010-11-25 07:26:38 +00:00
|
|
|
|
'nodestroy': True,
|
|
|
|
|
'context': {}
|
2010-10-01 07:54:12 +00:00
|
|
|
|
})
|
|
|
|
|
return action_data
|
2010-10-04 05:18:23 +00:00
|
|
|
|
|
2010-11-19 13:48:01 +00:00
|
|
|
|
def open_attachment(self, cr, uid, ids, context=None):
|
2010-10-01 07:54:12 +00:00
|
|
|
|
""" To Open attachments
|
|
|
|
|
@param self: The object pointer.
|
|
|
|
|
@param cr: A database cursor
|
|
|
|
|
@param uid: ID of the user currently logged in
|
|
|
|
|
@param ids: the ID of messages
|
|
|
|
|
@param context: A standard dictionary
|
|
|
|
|
"""
|
|
|
|
|
action_data = False
|
|
|
|
|
action_pool = self.pool.get('ir.actions.act_window')
|
2010-11-19 13:48:01 +00:00
|
|
|
|
message_pool = self.browse(cr ,uid, ids, context=context)[0]
|
2011-03-10 10:39:51 +00:00
|
|
|
|
att_ids = [x.id for x in message_pool.attachment_ids]
|
2010-10-01 07:54:12 +00:00
|
|
|
|
action_ids = action_pool.search(cr, uid, [('res_model', '=', 'ir.attachment')])
|
|
|
|
|
if action_ids:
|
|
|
|
|
action_data = action_pool.read(cr, uid, action_ids[0], context=context)
|
|
|
|
|
action_data.update({
|
2011-01-28 13:32:59 +00:00
|
|
|
|
'domain': [('id','in',att_ids)],
|
2010-10-01 07:54:12 +00:00
|
|
|
|
'nodestroy': True
|
|
|
|
|
})
|
|
|
|
|
return action_data
|
|
|
|
|
|
2010-09-09 11:40:21 +00:00
|
|
|
|
def truncate_data(self, cr, uid, data, context=None):
|
|
|
|
|
data_list = data and data.split('\n') or []
|
|
|
|
|
if len(data_list) > 3:
|
|
|
|
|
res = '\n\t'.join(data_list[:3]) + '...'
|
|
|
|
|
else:
|
|
|
|
|
res = '\n\t'.join(data_list)
|
|
|
|
|
return res
|
|
|
|
|
|
2010-07-16 10:41:21 +00:00
|
|
|
|
def _get_display_text(self, cr, uid, ids, name, arg, context=None):
|
2010-07-27 15:26:37 +00:00
|
|
|
|
if context is None:
|
|
|
|
|
context = {}
|
|
|
|
|
tz = context.get('tz')
|
2010-07-16 10:41:21 +00:00
|
|
|
|
result = {}
|
|
|
|
|
for message in self.browse(cr, uid, ids, context=context):
|
|
|
|
|
msg_txt = ''
|
|
|
|
|
if message.history:
|
2011-03-23 16:22:06 +00:00
|
|
|
|
msg_txt += _('%s wrote on %s: \n Subject: %s \n\t') % (message.email_from or '/', format_date_tz(message.date, tz), message.name)
|
2010-07-28 10:08:30 +00:00
|
|
|
|
if message.description:
|
2010-09-09 11:40:21 +00:00
|
|
|
|
msg_txt += self.truncate_data(cr, uid, message.description, context=context)
|
2010-07-16 10:41:21 +00:00
|
|
|
|
else:
|
2011-03-17 17:33:16 +00:00
|
|
|
|
msg_txt = _('%s on %s:\n\t') % (message.user_id.name or '/', format_date_tz(message.date, tz))
|
2011-02-08 13:36:40 +00:00
|
|
|
|
msg_txt += message.name
|
2010-07-16 10:41:21 +00:00
|
|
|
|
result[message.id] = msg_txt
|
|
|
|
|
return result
|
|
|
|
|
|
2010-04-29 13:49:19 +00:00
|
|
|
|
_name = 'mailgate.message'
|
|
|
|
|
_description = 'Mailgateway Message'
|
2010-07-16 10:41:21 +00:00
|
|
|
|
_order = 'date desc'
|
2010-04-29 13:49:19 +00:00
|
|
|
|
_columns = {
|
2010-07-14 11:42:48 +00:00
|
|
|
|
'name':fields.text('Subject', readonly=True),
|
|
|
|
|
'model': fields.char('Object Name', size=128, select=1, readonly=True),
|
|
|
|
|
'res_id': fields.integer('Resource ID', select=1, readonly=True),
|
2010-06-24 12:57:47 +00:00
|
|
|
|
'ref_id': fields.char('Reference Id', size=256, readonly=True, help="Message Id in Email Server.", select=True),
|
2010-07-14 11:42:48 +00:00
|
|
|
|
'date': fields.datetime('Date', readonly=True),
|
|
|
|
|
'history': fields.boolean('Is History?', readonly=True),
|
2010-06-29 09:10:06 +00:00
|
|
|
|
'user_id': fields.many2one('res.users', 'User Responsible', readonly=True),
|
2011-04-27 05:23:03 +00:00
|
|
|
|
'message': fields.text('Message', readonly=True),
|
2010-07-14 11:42:48 +00:00
|
|
|
|
'email_from': fields.char('From', size=128, help="Email From", readonly=True),
|
|
|
|
|
'email_to': fields.char('To', help="Email Recipients", size=256, readonly=True),
|
|
|
|
|
'email_cc': fields.char('Cc', help="Carbon Copy Email Recipients", size=256, readonly=True),
|
|
|
|
|
'email_bcc': fields.char('Bcc', help='Blind Carbon Copy Email Recipients', size=256, readonly=True),
|
2010-06-24 19:53:32 +00:00
|
|
|
|
'message_id': fields.char('Message Id', size=1024, readonly=True, help="Message Id on Email.", select=True),
|
2010-07-02 16:31:01 +00:00
|
|
|
|
'references': fields.text('References', readonly=True, help="References emails."),
|
2010-07-14 11:42:48 +00:00
|
|
|
|
'description': fields.text('Description', readonly=True),
|
2010-06-29 09:10:06 +00:00
|
|
|
|
'partner_id': fields.many2one('res.partner', 'Partner', required=False),
|
2010-07-14 11:42:48 +00:00
|
|
|
|
'attachment_ids': fields.many2many('ir.attachment', 'message_attachment_rel', 'message_id', 'attachment_id', 'Attachments', readonly=True),
|
2011-07-01 23:41:24 +00:00
|
|
|
|
'display_text': fields.function(_get_display_text, type='text', size="512", string='Display Text'),
|
2010-04-29 13:49:19 +00:00
|
|
|
|
}
|
|
|
|
|
|
2010-06-29 09:10:06 +00:00
|
|
|
|
def init(self, cr):
|
|
|
|
|
cr.execute("""SELECT indexname
|
|
|
|
|
FROM pg_indexes
|
|
|
|
|
WHERE indexname = 'mailgate_message_res_id_model_idx'""")
|
|
|
|
|
if not cr.fetchone():
|
|
|
|
|
cr.execute("""CREATE INDEX mailgate_message_res_id_model_idx
|
|
|
|
|
ON mailgate_message (model, res_id)""")
|
|
|
|
|
|
2010-04-29 13:49:19 +00:00
|
|
|
|
mailgate_message()
|
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
class mailgate_tool(osv.osv_memory):
|
|
|
|
|
|
|
|
|
|
_name = 'email.server.tools'
|
|
|
|
|
_description = "Email Server Tools"
|
2010-06-25 17:41:41 +00:00
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
def _decode_header(self, text):
|
2010-07-02 13:44:45 +00:00
|
|
|
|
"""Returns unicode() string conversion of the the given encoded smtp header"""
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if text:
|
2010-07-02 13:44:45 +00:00
|
|
|
|
text = decode_header(text.replace('\r', ''))
|
|
|
|
|
return ''.join([tools.ustr(x[0], x[1]) for x in text])
|
2010-06-25 17:41:41 +00:00
|
|
|
|
|
2010-06-25 22:02:47 +00:00
|
|
|
|
def to_email(self,text):
|
|
|
|
|
return re.findall(r'([^ ,<@]+@[^> ,]+)',text)
|
2010-06-25 17:41:41 +00:00
|
|
|
|
|
2010-06-25 09:57:41 +00:00
|
|
|
|
def history(self, cr, uid, model, res_ids, msg, attach, context=None):
|
2010-06-24 19:53:32 +00:00
|
|
|
|
"""This function creates history for mails fetched
|
|
|
|
|
@param self: The object pointer
|
|
|
|
|
@param cr: the current row, from the database cursor,
|
|
|
|
|
@param uid: the current user’s ID for security checks,
|
|
|
|
|
@param model: OpenObject Model
|
2010-08-13 12:20:05 +00:00
|
|
|
|
@param res_ids: Ids of the record of OpenObject model created
|
2010-06-24 19:53:32 +00:00
|
|
|
|
@param msg: Email details
|
|
|
|
|
@param attach: Email attachments
|
|
|
|
|
"""
|
|
|
|
|
if isinstance(res_ids, (int, long)):
|
|
|
|
|
res_ids = [res_ids]
|
|
|
|
|
|
|
|
|
|
msg_pool = self.pool.get('mailgate.message')
|
|
|
|
|
for res_id in res_ids:
|
2010-09-08 12:34:08 +00:00
|
|
|
|
case = self.pool.get(model).browse(cr, uid, res_id, context=context)
|
|
|
|
|
partner_id = hasattr(case, 'partner_id') and (case.partner_id and case.partner_id.id or False) or False
|
|
|
|
|
if not partner_id and model == 'res.partner':
|
|
|
|
|
partner_id = res_id
|
2010-06-24 19:53:32 +00:00
|
|
|
|
msg_data = {
|
2010-08-13 12:20:05 +00:00
|
|
|
|
'name': msg.get('subject', 'No subject'),
|
|
|
|
|
'date': msg.get('date'),
|
|
|
|
|
'description': msg.get('body', msg.get('from')),
|
|
|
|
|
'history': True,
|
2010-09-08 12:34:08 +00:00
|
|
|
|
'partner_id': partner_id,
|
2010-09-14 10:19:00 +00:00
|
|
|
|
'model': model,
|
2010-08-13 12:20:05 +00:00
|
|
|
|
'email_cc': msg.get('cc'),
|
|
|
|
|
'email_from': msg.get('from'),
|
|
|
|
|
'email_to': msg.get('to'),
|
|
|
|
|
'message_id': msg.get('message-id'),
|
|
|
|
|
'references': msg.get('references') or msg.get('in-reply-to'),
|
|
|
|
|
'res_id': res_id,
|
|
|
|
|
'user_id': uid,
|
|
|
|
|
'attachment_ids': [(6, 0, attach)]
|
2010-06-24 19:53:32 +00:00
|
|
|
|
}
|
2010-06-29 09:10:06 +00:00
|
|
|
|
msg_pool.create(cr, uid, msg_data, context=context)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
return True
|
2010-06-25 17:41:41 +00:00
|
|
|
|
|
2010-06-28 10:38:32 +00:00
|
|
|
|
def email_forward(self, cr, uid, model, res_ids, msg, email_error=False, context=None):
|
2010-06-25 22:02:47 +00:00
|
|
|
|
"""Sends an email to all people following the thread
|
2010-06-28 10:38:32 +00:00
|
|
|
|
@param res_id: Id of the record of OpenObject model created from the email message
|
|
|
|
|
@param msg: email.message.Message to forward
|
2010-06-25 22:02:47 +00:00
|
|
|
|
@param email_error: Default Email address in case of any Problem
|
2010-06-24 19:53:32 +00:00
|
|
|
|
"""
|
2010-06-28 10:38:32 +00:00
|
|
|
|
model_pool = self.pool.get(model)
|
2010-06-25 22:02:47 +00:00
|
|
|
|
|
2010-06-28 10:38:32 +00:00
|
|
|
|
for res in model_pool.browse(cr, uid, res_ids, context=context):
|
|
|
|
|
message_followers = model_pool.message_followers(cr, uid, [res.id])[res.id]
|
2010-08-17 19:22:41 +00:00
|
|
|
|
message_followers_emails = self.to_email(','.join(filter(None, message_followers)))
|
2010-07-02 13:44:45 +00:00
|
|
|
|
message_recipients = self.to_email(','.join(filter(None,
|
|
|
|
|
[self._decode_header(msg['from']),
|
2010-06-28 10:38:32 +00:00
|
|
|
|
self._decode_header(msg['to']),
|
2010-07-02 13:44:45 +00:00
|
|
|
|
self._decode_header(msg['cc'])])))
|
2010-06-25 22:02:47 +00:00
|
|
|
|
message_forward = [i for i in message_followers_emails if (i and (i not in message_recipients))]
|
|
|
|
|
|
2010-06-28 10:38:32 +00:00
|
|
|
|
if message_forward:
|
|
|
|
|
# TODO: we need an interface for this for all types of objects, not just leads
|
2010-06-28 14:47:26 +00:00
|
|
|
|
if hasattr(res, 'section_id'):
|
2010-06-28 10:38:32 +00:00
|
|
|
|
del msg['reply-to']
|
2010-06-28 14:47:26 +00:00
|
|
|
|
msg['reply-to'] = res.section_id.reply_to
|
2010-06-28 10:38:32 +00:00
|
|
|
|
|
2010-06-28 14:47:26 +00:00
|
|
|
|
smtp_from = self.to_email(msg['from'])
|
|
|
|
|
if not tools.misc._email_send(smtp_from, message_forward, msg, openobject_id=res.id) and email_error:
|
2010-06-28 10:38:32 +00:00
|
|
|
|
subj = msg['subject']
|
|
|
|
|
del msg['subject'], msg['to'], msg['cc'], msg['bcc']
|
|
|
|
|
msg['subject'] = '[OpenERP-Forward-Failed] %s' % subj
|
|
|
|
|
msg['to'] = email_error
|
2010-06-28 14:47:26 +00:00
|
|
|
|
tools.misc._email_send(smtp_from, self.to_email(email_error), msg, openobject_id=res.id)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-09-30 15:30:03 +00:00
|
|
|
|
def process_email(self, cr, uid, model, message, custom_values=None, attach=True, context=None):
|
2010-06-28 14:47:26 +00:00
|
|
|
|
"""This function Processes email and create record for given OpenERP model
|
2010-06-24 19:53:32 +00:00
|
|
|
|
@param self: The object pointer
|
|
|
|
|
@param cr: the current row, from the database cursor,
|
|
|
|
|
@param uid: the current user’s ID for security checks,
|
|
|
|
|
@param model: OpenObject Model
|
2010-07-02 13:44:45 +00:00
|
|
|
|
@param message: Email details, passed as a string or an xmlrpclib.Binary
|
2010-06-25 12:18:47 +00:00
|
|
|
|
@param attach: Email attachments
|
2010-06-24 19:53:32 +00:00
|
|
|
|
@param context: A standard dictionary for contextual values"""
|
2010-07-02 13:44:45 +00:00
|
|
|
|
|
|
|
|
|
# extract message bytes, we are forced to pass the message as binary because
|
|
|
|
|
# we don't know its encoding until we parse its headers and hence can't
|
|
|
|
|
# convert it to utf-8 for transport between the mailgate script and here.
|
|
|
|
|
if isinstance(message, xmlrpclib.Binary):
|
|
|
|
|
message = str(message.data)
|
|
|
|
|
|
2010-11-23 07:05:05 +00:00
|
|
|
|
if context is None:
|
2010-06-24 19:53:32 +00:00
|
|
|
|
context = {}
|
2010-07-02 13:44:45 +00:00
|
|
|
|
|
2010-09-30 15:30:03 +00:00
|
|
|
|
if custom_values is None or not isinstance(custom_values, dict):
|
|
|
|
|
custom_values = {}
|
|
|
|
|
|
2010-07-02 13:44:45 +00:00
|
|
|
|
model_pool = self.pool.get(model)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
res_id = False
|
2010-07-02 13:44:45 +00:00
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
# Create New Record into particular model
|
2010-06-25 12:18:47 +00:00
|
|
|
|
def create_record(msg):
|
2010-06-29 13:56:11 +00:00
|
|
|
|
att_ids = []
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if hasattr(model_pool, 'message_new'):
|
2010-11-19 13:48:01 +00:00
|
|
|
|
res_id = model_pool.message_new(cr, uid, msg, context=context)
|
2010-09-30 15:30:03 +00:00
|
|
|
|
if custom_values:
|
|
|
|
|
model_pool.write(cr, uid, [res_id], custom_values, context=context)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
else:
|
|
|
|
|
data = {
|
2010-06-25 12:18:47 +00:00
|
|
|
|
'name': msg.get('subject'),
|
|
|
|
|
'email_from': msg.get('from'),
|
|
|
|
|
'email_cc': msg.get('cc'),
|
|
|
|
|
'user_id': False,
|
|
|
|
|
'description': msg.get('body'),
|
|
|
|
|
'state' : 'draft',
|
2010-06-24 19:53:32 +00:00
|
|
|
|
}
|
|
|
|
|
data.update(self.get_partner(cr, uid, msg.get('from'), context=context))
|
|
|
|
|
res_id = model_pool.create(cr, uid, data, context=context)
|
|
|
|
|
|
|
|
|
|
if attach:
|
|
|
|
|
for attachment in msg.get('attachments', []):
|
|
|
|
|
data_attach = {
|
2010-06-25 12:18:47 +00:00
|
|
|
|
'name': attachment,
|
|
|
|
|
'datas': binascii.b2a_base64(str(attachments.get(attachment))),
|
|
|
|
|
'datas_fname': attachment,
|
|
|
|
|
'description': 'Mail attachment',
|
|
|
|
|
'res_model': model,
|
|
|
|
|
'res_id': res_id,
|
2010-06-24 19:53:32 +00:00
|
|
|
|
}
|
2010-06-25 12:18:47 +00:00
|
|
|
|
att_ids.append(self.pool.get('ir.attachment').create(cr, uid, data_attach))
|
|
|
|
|
|
2010-06-29 09:10:06 +00:00
|
|
|
|
return res_id, att_ids
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-06-25 12:18:47 +00:00
|
|
|
|
# Warning: message_from_string doesn't always work correctly on unicode,
|
|
|
|
|
# we must use utf-8 strings here :-(
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if isinstance(message, unicode):
|
|
|
|
|
message = message.encode('utf-8')
|
|
|
|
|
msg_txt = email.message_from_string(message)
|
|
|
|
|
message_id = msg_txt.get('message-id', False)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
msg = {}
|
2010-06-25 12:18:47 +00:00
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if not message_id:
|
2010-06-25 12:18:47 +00:00
|
|
|
|
# Very unusual situation, be we should be fault-tolerant here
|
|
|
|
|
message_id = time.time()
|
2010-07-02 13:44:45 +00:00
|
|
|
|
msg_txt['message-id'] = message_id
|
2010-06-25 12:18:47 +00:00
|
|
|
|
_logger.info('Message without message-id, generating a random one: %s', message_id)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
|
|
|
|
fields = msg_txt.keys()
|
|
|
|
|
msg['id'] = message_id
|
2010-06-25 12:18:47 +00:00
|
|
|
|
msg['message-id'] = message_id
|
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if 'Subject' in fields:
|
|
|
|
|
msg['subject'] = self._decode_header(msg_txt.get('Subject'))
|
|
|
|
|
|
|
|
|
|
if 'Content-Type' in fields:
|
|
|
|
|
msg['content-type'] = msg_txt.get('Content-Type')
|
|
|
|
|
|
|
|
|
|
if 'From' in fields:
|
|
|
|
|
msg['from'] = self._decode_header(msg_txt.get('From'))
|
|
|
|
|
|
|
|
|
|
if 'Delivered-To' in fields:
|
|
|
|
|
msg['to'] = self._decode_header(msg_txt.get('Delivered-To'))
|
|
|
|
|
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if 'CC' in fields:
|
|
|
|
|
msg['cc'] = self._decode_header(msg_txt.get('CC'))
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if 'Reply-to' in fields:
|
2010-06-24 19:53:32 +00:00
|
|
|
|
msg['reply'] = self._decode_header(msg_txt.get('Reply-To'))
|
|
|
|
|
|
|
|
|
|
if 'Date' in fields:
|
2010-07-02 13:44:45 +00:00
|
|
|
|
msg['date'] = self._decode_header(msg_txt.get('Date'))
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
|
|
|
|
if 'Content-Transfer-Encoding' in fields:
|
|
|
|
|
msg['encoding'] = msg_txt.get('Content-Transfer-Encoding')
|
|
|
|
|
|
|
|
|
|
if 'References' in fields:
|
2010-06-25 12:18:47 +00:00
|
|
|
|
msg['references'] = msg_txt.get('References')
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-08-05 09:19:09 +00:00
|
|
|
|
if 'In-Reply-To' in fields:
|
|
|
|
|
msg['in-reply-to'] = msg_txt.get('In-Reply-To')
|
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if 'X-Priority' in fields:
|
2010-07-02 13:44:45 +00:00
|
|
|
|
msg['priority'] = msg_txt.get('X-Priority', '3 (Normal)').split(' ')[0]
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if not msg_txt.is_multipart() or 'text/plain' in msg.get('Content-Type', ''):
|
2010-06-24 19:53:32 +00:00
|
|
|
|
encoding = msg_txt.get_content_charset()
|
2010-07-02 13:44:45 +00:00
|
|
|
|
body = msg_txt.get_payload(decode=True)
|
2010-11-11 10:14:26 +00:00
|
|
|
|
if 'text/html' in msg_txt.get('Content-Type', ''):
|
|
|
|
|
body = tools.html2plaintext(body)
|
2010-07-02 13:44:45 +00:00
|
|
|
|
msg['body'] = tools.ustr(body, encoding)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
|
|
|
|
attachments = {}
|
2010-07-02 13:44:45 +00:00
|
|
|
|
has_plain_text = False
|
2010-06-25 12:18:47 +00:00
|
|
|
|
if msg_txt.is_multipart() or 'multipart/alternative' in msg.get('content-type', ''):
|
2010-06-24 19:53:32 +00:00
|
|
|
|
body = ""
|
|
|
|
|
for part in msg_txt.walk():
|
|
|
|
|
if part.get_content_maintype() == 'multipart':
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
|
|
encoding = part.get_content_charset()
|
2010-07-02 13:44:45 +00:00
|
|
|
|
filename = part.get_filename()
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if part.get_content_maintype()=='text':
|
|
|
|
|
content = part.get_payload(decode=True)
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if filename:
|
2010-06-24 19:53:32 +00:00
|
|
|
|
attachments[filename] = content
|
2010-07-02 13:44:45 +00:00
|
|
|
|
elif not has_plain_text:
|
|
|
|
|
# main content parts should have 'text' maintype
|
|
|
|
|
# and no filename. we ignore the html part if
|
|
|
|
|
# there is already a plaintext part without filename,
|
|
|
|
|
# because presumably these are alternatives.
|
|
|
|
|
content = tools.ustr(content, encoding)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if part.get_content_subtype() == 'html':
|
2010-06-29 13:56:11 +00:00
|
|
|
|
body = tools.ustr(tools.html2plaintext(content))
|
2010-06-24 19:53:32 +00:00
|
|
|
|
elif part.get_content_subtype() == 'plain':
|
2010-06-30 18:11:48 +00:00
|
|
|
|
body = content
|
2010-07-02 13:44:45 +00:00
|
|
|
|
has_plain_text = True
|
|
|
|
|
elif part.get_content_maintype() in ('application', 'image'):
|
2011-06-16 12:34:36 +00:00
|
|
|
|
if filename and attach:
|
2010-06-24 19:53:32 +00:00
|
|
|
|
attachments[filename] = part.get_payload(decode=True)
|
|
|
|
|
else:
|
|
|
|
|
res = part.get_payload(decode=True)
|
2010-07-02 13:44:45 +00:00
|
|
|
|
body += tools.ustr(res, encoding)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
|
|
|
|
|
msg['body'] = body
|
|
|
|
|
msg['attachments'] = attachments
|
|
|
|
|
res_ids = []
|
2010-06-29 09:10:06 +00:00
|
|
|
|
attachment_ids = []
|
2010-06-24 19:53:32 +00:00
|
|
|
|
new_res_id = False
|
2010-08-05 09:19:09 +00:00
|
|
|
|
if msg.get('references') or msg.get('in-reply-to'):
|
|
|
|
|
references = msg.get('references') or msg.get('in-reply-to')
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if '\r\n' in references:
|
2010-08-05 09:19:09 +00:00
|
|
|
|
references = references.split('\r\n')
|
2010-06-24 19:53:32 +00:00
|
|
|
|
else:
|
2010-08-05 09:19:09 +00:00
|
|
|
|
references = references.split(' ')
|
2010-06-24 19:53:32 +00:00
|
|
|
|
for ref in references:
|
|
|
|
|
ref = ref.strip()
|
|
|
|
|
res_id = tools.misc.reference_re.search(ref)
|
|
|
|
|
if res_id:
|
|
|
|
|
res_id = res_id.group(1)
|
|
|
|
|
else:
|
|
|
|
|
res_id = tools.misc.res_re.search(msg['subject'])
|
|
|
|
|
if res_id:
|
|
|
|
|
res_id = res_id.group(1)
|
|
|
|
|
if res_id:
|
|
|
|
|
res_id = int(res_id)
|
|
|
|
|
model_pool = self.pool.get(model)
|
2010-07-02 13:44:45 +00:00
|
|
|
|
if model_pool.exists(cr, uid, res_id):
|
|
|
|
|
res_ids.append(res_id)
|
|
|
|
|
if hasattr(model_pool, 'message_update'):
|
|
|
|
|
model_pool.message_update(cr, uid, [res_id], {}, msg, context=context)
|
|
|
|
|
else:
|
|
|
|
|
raise NotImplementedError('model %s does not support updating records, mailgate API method message_update() is missing'%model)
|
2010-06-25 12:18:47 +00:00
|
|
|
|
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if not len(res_ids):
|
2010-06-29 09:10:06 +00:00
|
|
|
|
new_res_id, attachment_ids = create_record(msg)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
res_ids = [new_res_id]
|
2010-07-02 13:44:45 +00:00
|
|
|
|
|
2010-06-25 09:57:41 +00:00
|
|
|
|
# Store messages
|
2010-06-25 12:18:47 +00:00
|
|
|
|
context.update({'model' : model})
|
2010-07-02 16:31:01 +00:00
|
|
|
|
if hasattr(model_pool, 'history'):
|
|
|
|
|
model_pool.history(cr, uid, res_ids, _('receive'), history=True,
|
2010-07-02 05:38:19 +00:00
|
|
|
|
subject = msg.get('subject'),
|
|
|
|
|
email = msg.get('to'),
|
|
|
|
|
details = msg.get('body'),
|
|
|
|
|
email_from = msg.get('from'),
|
2010-07-02 16:31:01 +00:00
|
|
|
|
email_cc = msg.get('cc'),
|
2010-07-02 05:38:19 +00:00
|
|
|
|
message_id = msg.get('message-id'),
|
2010-08-05 09:19:09 +00:00
|
|
|
|
references = msg.get('references', False) or msg.get('in-reply-to', False),
|
2010-07-02 05:38:19 +00:00
|
|
|
|
attach = attachments.items(),
|
2010-11-19 09:22:25 +00:00
|
|
|
|
email_date = msg.get('date'),
|
2010-07-02 05:38:19 +00:00
|
|
|
|
context = context)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
else:
|
2010-06-29 09:10:06 +00:00
|
|
|
|
self.history(cr, uid, model, res_ids, msg, attachment_ids, context=context)
|
2010-06-28 10:38:32 +00:00
|
|
|
|
self.email_forward(cr, uid, model, res_ids, msg_txt)
|
2010-06-24 19:53:32 +00:00
|
|
|
|
return new_res_id
|
|
|
|
|
|
|
|
|
|
def get_partner(self, cr, uid, from_email, context=None):
|
|
|
|
|
"""This function returns partner Id based on email passed
|
|
|
|
|
@param self: The object pointer
|
|
|
|
|
@param cr: the current row, from the database cursor,
|
|
|
|
|
@param uid: the current user’s ID for security checks
|
|
|
|
|
@param from_email: email address based on that function will search for the correct
|
|
|
|
|
"""
|
|
|
|
|
address_pool = self.pool.get('res.partner.address')
|
|
|
|
|
res = {
|
2010-06-25 12:18:47 +00:00
|
|
|
|
'partner_address_id': False,
|
2010-06-24 19:53:32 +00:00
|
|
|
|
'partner_id': False
|
|
|
|
|
}
|
2010-06-25 22:02:47 +00:00
|
|
|
|
from_email = self.to_email(from_email)[0]
|
2010-07-19 10:06:40 +00:00
|
|
|
|
address_ids = address_pool.search(cr, uid, [('email', 'like', from_email)])
|
2010-06-24 19:53:32 +00:00
|
|
|
|
if address_ids:
|
|
|
|
|
address = address_pool.browse(cr, uid, address_ids[0])
|
|
|
|
|
res['partner_address_id'] = address_ids[0]
|
|
|
|
|
res['partner_id'] = address.partner_id.id
|
|
|
|
|
|
|
|
|
|
return res
|
|
|
|
|
|
|
|
|
|
mailgate_tool()
|
2010-09-29 09:49:50 +00:00
|
|
|
|
|
2010-10-01 07:54:12 +00:00
|
|
|
|
# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4:
|