mirror of https://github.com/OCA/social.git
[ADD] mass_mailing_sending_queue addon
parent
55250b99d4
commit
29fae520a4
|
@ -0,0 +1,93 @@
|
||||||
|
.. image:: https://img.shields.io/badge/licence-AGPL--3-blue.svg
|
||||||
|
:target: http://www.gnu.org/licenses/agpl-3.0-standalone.html
|
||||||
|
:alt: License: AGPL-3
|
||||||
|
|
||||||
|
==========================
|
||||||
|
Mass mailing sending queue
|
||||||
|
==========================
|
||||||
|
|
||||||
|
This module adds a queue for generating mail records when mass mailing
|
||||||
|
'Send to All' button is clicked. This is an additional queue, apart from
|
||||||
|
the existing one (implemented in addons/mail) for doing the actual sending.
|
||||||
|
|
||||||
|
|
||||||
|
Configuration
|
||||||
|
=============
|
||||||
|
|
||||||
|
There is a system parameter, 'mail.mass_mailing.sending.batch_size'
|
||||||
|
(default value is 500), to control how many emails are created in each
|
||||||
|
cron iteration (method 'mail.mass_mailing.sending.cron()').
|
||||||
|
|
||||||
|
|
||||||
|
Usage
|
||||||
|
=====
|
||||||
|
|
||||||
|
Without this module, when 'Send to All' button is clicked at mass mailing form,
|
||||||
|
all 'mail.mail' and 'mail.mail.statistics' objects are created. This process
|
||||||
|
might take a long time if the recipient list is 10k+ and the famous
|
||||||
|
"Take a minute to get a coffee, because it's loading..." text might appear.
|
||||||
|
|
||||||
|
With this new queue, mass mailing will appear in 'Sending' state to the user
|
||||||
|
until all emails are sent or failed. After 'Send to All' button is clicked,
|
||||||
|
the user will quickly land to the mass mailing form.
|
||||||
|
|
||||||
|
In 'Mass mailing' form, a new tab "Sending tasks" has been added where the
|
||||||
|
user can check the Sent mails history.
|
||||||
|
|
||||||
|
In 'Settings > Technical > Email > Mass mailing sending' allowed users can
|
||||||
|
track all running mass mailing sending objects and see:
|
||||||
|
|
||||||
|
* Pending recipients: Number of recipients for which the email is not yet created.
|
||||||
|
* Start date: Date when user press 'Send to All' button.
|
||||||
|
* Mails to be sent: number of emails waiting to be sent.
|
||||||
|
* Sent mails: number of emails successfully sent.
|
||||||
|
* Failed mails: number of unsent emails due to error.
|
||||||
|
|
||||||
|
NOTE: User will not be able to send the same mass mailing again if another
|
||||||
|
one is ongoing. An UserError exception is raised in this case.
|
||||||
|
|
||||||
|
NOTE: If number of recipients are less than 'batch_size / 2', then all
|
||||||
|
emails are created when 'Send to All' button is clicked (standard way).
|
||||||
|
Although a sending object is created anyway in order to be consistent.
|
||||||
|
|
||||||
|
.. image:: https://odoo-community.org/website/image/ir.attachment/5784_f2813bd/datas
|
||||||
|
:alt: Try me on Runbot
|
||||||
|
:target: https://runbot.odoo-community.org/runbot/205/8.0
|
||||||
|
|
||||||
|
|
||||||
|
Bug Tracker
|
||||||
|
===========
|
||||||
|
|
||||||
|
Bugs are tracked on `GitHub Issues
|
||||||
|
<https://github.com/OCA/social/issues>`_. In case of trouble, please
|
||||||
|
check there if your issue has already been reported. If you spotted it first,
|
||||||
|
help us smashing it by providing a detailed and welcomed feedback.
|
||||||
|
|
||||||
|
Credits
|
||||||
|
=======
|
||||||
|
|
||||||
|
Images
|
||||||
|
------
|
||||||
|
|
||||||
|
* Odoo Community Association: `Icon <https://github.com/OCA/maintainer-tools/blob/master/template/module/static/description/icon.svg>`_.
|
||||||
|
|
||||||
|
Contributors
|
||||||
|
------------
|
||||||
|
|
||||||
|
* Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
* Pedro M. Baeza <pedro.baeza@tecnativa.com>
|
||||||
|
|
||||||
|
Maintainer
|
||||||
|
----------
|
||||||
|
|
||||||
|
.. image:: https://odoo-community.org/logo.png
|
||||||
|
:alt: Odoo Community Association
|
||||||
|
:target: https://odoo-community.org
|
||||||
|
|
||||||
|
This module is maintained by the OCA.
|
||||||
|
|
||||||
|
OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||||
|
mission is to support the collaborative development of Odoo features and
|
||||||
|
promote its widespread use.
|
||||||
|
|
||||||
|
To contribute to this module, please visit https://odoo-community.org.
|
|
@ -0,0 +1,4 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import models
|
|
@ -0,0 +1,24 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
{
|
||||||
|
"name": "Mass mailing sending queue",
|
||||||
|
"summary": "A new queue for sending mass mailing",
|
||||||
|
"version": "8.0.1.0.0",
|
||||||
|
"category": "Marketing",
|
||||||
|
"website": "https://odoo-community.org/",
|
||||||
|
"author": "Tecnativa, Odoo Community Association (OCA)",
|
||||||
|
"license": "AGPL-3",
|
||||||
|
"application": False,
|
||||||
|
"installable": True,
|
||||||
|
"depends": [
|
||||||
|
"mass_mailing",
|
||||||
|
],
|
||||||
|
"data": [
|
||||||
|
"data/ir_config_parameter.xml",
|
||||||
|
"data/ir_cron.xml",
|
||||||
|
"security/ir.model.access.csv",
|
||||||
|
"views/mass_mailing_sending_view.xml",
|
||||||
|
"views/mass_mailing_view.xml",
|
||||||
|
],
|
||||||
|
}
|
|
@ -0,0 +1,15 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||||
|
|
||||||
|
<openerp>
|
||||||
|
<data noupdate="1">
|
||||||
|
|
||||||
|
<record id="sending_batch_size" model="ir.config_parameter" forcecreate="True">
|
||||||
|
<field name="key">mail.mass_mailing.sending.batch_size</field>
|
||||||
|
<field name="value">500</field>
|
||||||
|
<field name="group_ids" eval="[(6, False, [ref('base.group_system')])]" />
|
||||||
|
</record>
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</openerp>
|
|
@ -0,0 +1,19 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||||
|
|
||||||
|
<openerp>
|
||||||
|
<data noupdate="1">
|
||||||
|
|
||||||
|
<record id="cronjob" model="ir.cron">
|
||||||
|
<field name="name">Mass mailing sending queue</field>
|
||||||
|
<field name="interval_type">minutes</field>
|
||||||
|
<field name="interval_number">1</field>
|
||||||
|
<field name="numbercall">-1</field>
|
||||||
|
<field name="model">mail.mass_mailing.sending</field>
|
||||||
|
<field name="function">cron</field>
|
||||||
|
<field name="nextcall">2016-01-01</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</openerp>
|
|
@ -0,0 +1,156 @@
|
||||||
|
# Translation of Odoo Server.
|
||||||
|
# This file contains the translation of the following modules:
|
||||||
|
# * mass_mailing_sending_queue
|
||||||
|
#
|
||||||
|
msgid ""
|
||||||
|
msgstr ""
|
||||||
|
"Project-Id-Version: Odoo Server 8.0\n"
|
||||||
|
"Report-Msgid-Bugs-To: \n"
|
||||||
|
"POT-Creation-Date: 2016-09-08 13:56+0000\n"
|
||||||
|
"PO-Revision-Date: 2016-09-08 13:56+0000\n"
|
||||||
|
"Last-Translator: <>\n"
|
||||||
|
"Language-Team: \n"
|
||||||
|
"MIME-Version: 1.0\n"
|
||||||
|
"Content-Type: text/plain; charset=UTF-8\n"
|
||||||
|
"Content-Transfer-Encoding: \n"
|
||||||
|
"Plural-Forms: \n"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,create_uid:0
|
||||||
|
msgid "Created by"
|
||||||
|
msgstr "Creado por"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,create_date:0
|
||||||
|
msgid "Created on"
|
||||||
|
msgstr "Creado en"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,display_name:0
|
||||||
|
msgid "Display Name"
|
||||||
|
msgstr "Nombre mostrado"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: model:ir.model,name:mass_mailing_sending_queue.model_mail_mail_statistics
|
||||||
|
msgid "Email Statistics"
|
||||||
|
msgstr "Estadísticas de correo electrónico"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,failed:0
|
||||||
|
msgid "Emails failed"
|
||||||
|
msgstr "Correos electrónicos fallidos"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,sending:0
|
||||||
|
msgid "Emails sending"
|
||||||
|
msgstr "Emails enviándose"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,sent:0
|
||||||
|
msgid "Emails sent"
|
||||||
|
msgstr "Emails enviados"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
#: selection:mail.mass_mailing.sending,state:0
|
||||||
|
msgid "Enqueued"
|
||||||
|
msgstr "En cola"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
msgid "Group By"
|
||||||
|
msgstr "Agrupar por"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,id:0
|
||||||
|
msgid "ID"
|
||||||
|
msgstr "ID"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,__last_update:0
|
||||||
|
msgid "Last Modified on"
|
||||||
|
msgstr "Última modficación en"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,write_uid:0
|
||||||
|
msgid "Last Updated by"
|
||||||
|
msgstr "Última modficación por"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,write_date:0
|
||||||
|
msgid "Last Updated on"
|
||||||
|
msgstr "Última actualización en"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: model:ir.model,name:mass_mailing_sending_queue.model_mail_mass_mailing
|
||||||
|
msgid "Mass Mailing"
|
||||||
|
msgstr "Envío masivo"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
#: field:mail.mass_mailing.sending,mass_mailing_id:0
|
||||||
|
msgid "Mass mailing"
|
||||||
|
msgstr "Envío masivo"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: model:ir.ui.menu,name:mass_mailing_sending_queue.menu_mass_mailing_sending
|
||||||
|
#: field:mail.mail.statistics,mass_mailing_sending_id:0
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_form
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
msgid "Mass mailing sending"
|
||||||
|
msgstr "Envío de correo masivo"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: model:ir.actions.act_window,name:mass_mailing_sending_queue.action_view_mail_mass_mailing_sending
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_tree
|
||||||
|
msgid "Mass mailing sendings"
|
||||||
|
msgstr "Envíos de correo masivo"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing,pending:0
|
||||||
|
msgid "Pending"
|
||||||
|
msgstr "Pendiente"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: code:addons/mass_mailing_sending_queue/models/mail_mass_mailing.py:23
|
||||||
|
#: code:addons/mass_mailing_sending_queue/models/mail_mass_mailing_sending.py:67
|
||||||
|
#, python-format
|
||||||
|
msgid "Please select recipients."
|
||||||
|
msgstr "Por favor, seleccione algún destinatario"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,pending:0
|
||||||
|
msgid "Recipients pending"
|
||||||
|
msgstr "Destinatarios pendientes"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
msgid "Running"
|
||||||
|
msgstr "Ejecutándose"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
#: selection:mail.mass_mailing.sending,state:0
|
||||||
|
msgid "Sending"
|
||||||
|
msgstr "Enviando"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing.sending:mass_mailing_sending_queue.view_mail_mass_mailing_sending_search
|
||||||
|
#: selection:mail.mass_mailing.sending,state:0
|
||||||
|
msgid "Sent"
|
||||||
|
msgstr "Enviado"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: field:mail.mass_mailing.sending,state:0
|
||||||
|
msgid "State"
|
||||||
|
msgstr "Estado"
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing:mass_mailing_sending_queue.view_mail_mass_mailing_form
|
||||||
|
msgid "recipients pending for creating email."
|
||||||
|
msgstr "destinatarios pendientes de crear el correo electrónico."
|
||||||
|
|
||||||
|
#. module: mass_mailing_sending_queue
|
||||||
|
#: view:mail.mass_mailing:mass_mailing_sending_queue.view_mail_mass_mailing_form
|
||||||
|
msgid "{'invisible': [('scheduled', '=', 0), ('pending', '=', 0)]}"
|
||||||
|
msgstr "{'invisible': [('scheduled', '=', 0), ('pending', '=', 0)]}"
|
|
@ -0,0 +1,8 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import mail_mail_statistics
|
||||||
|
from . import mail_mass_mailing
|
||||||
|
from . import mail_mass_mailing_sending
|
||||||
|
from . import mail_mail
|
|
@ -0,0 +1,20 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# © 2016 Antonio Espinosa - <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from openerp import models, api
|
||||||
|
|
||||||
|
|
||||||
|
class MailMail(models.Model):
|
||||||
|
_inherit = 'mail.mail'
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def _postprocess_sent_message(self, mail, mail_sent=True):
|
||||||
|
# Read before super, because mail will be removed if sent successfully
|
||||||
|
stats = mail.statistics_ids
|
||||||
|
res = super(MailMail, self)._postprocess_sent_message(
|
||||||
|
mail, mail_sent=mail_sent)
|
||||||
|
for stat in stats.filtered(
|
||||||
|
lambda r: r.mass_mailing_sending_id.state == 'sending'):
|
||||||
|
stat.mass_mailing_sending_id._process_sending()
|
||||||
|
return res
|
|
@ -0,0 +1,20 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# © 2016 Antonio Espinosa - <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||||
|
|
||||||
|
from openerp import api, fields, models
|
||||||
|
|
||||||
|
|
||||||
|
class MailMailStatistics(models.Model):
|
||||||
|
_inherit = 'mail.mail.statistics'
|
||||||
|
|
||||||
|
mass_mailing_sending_id = fields.Many2one(
|
||||||
|
comodel_name='mail.mass_mailing.sending',
|
||||||
|
string="Mass mailing sending", readonly=True)
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def create(self, vals):
|
||||||
|
res = super(MailMailStatistics, self).create(vals)
|
||||||
|
res.mass_mailing_sending_id = \
|
||||||
|
self.env.context.get('mass_mailing_sending_id', False)
|
||||||
|
return res
|
|
@ -0,0 +1,103 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from openerp import api, fields, models, _
|
||||||
|
from openerp.exceptions import Warning as UserError
|
||||||
|
|
||||||
|
|
||||||
|
class MailMassMailing(models.Model):
|
||||||
|
_inherit = 'mail.mass_mailing'
|
||||||
|
|
||||||
|
state = fields.Selection(selection_add=[
|
||||||
|
('sending', "Sending"),
|
||||||
|
])
|
||||||
|
pending_count = fields.Integer(
|
||||||
|
string="Pending", compute='_compute_pending_count')
|
||||||
|
mass_mailing_sending_ids = fields.One2many(
|
||||||
|
comodel_name='mail.mass_mailing.sending', readonly=True,
|
||||||
|
inverse_name='mass_mailing_id', string="Sending tasks")
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def read_group(self, domain, fields, groupby, **kwargs):
|
||||||
|
# Add 'sending' state group, even if no results.
|
||||||
|
# This is needed for kanban view, columns are showed always
|
||||||
|
res = super(MailMassMailing, self).read_group(
|
||||||
|
domain, fields, groupby, **kwargs)
|
||||||
|
if groupby and groupby[0] == "state":
|
||||||
|
group_domain = domain + [('state', '=', 'sending')]
|
||||||
|
count = self.search_count(group_domain)
|
||||||
|
res.append({
|
||||||
|
'__context': {'group_by': groupby[1:]},
|
||||||
|
'__domain': group_domain,
|
||||||
|
'state': ('sending', _("Sending")),
|
||||||
|
'state_count': count,
|
||||||
|
})
|
||||||
|
return res
|
||||||
|
|
||||||
|
def _sendings_get(self):
|
||||||
|
self.ensure_one()
|
||||||
|
return self.env['mail.mass_mailing.sending'].search([
|
||||||
|
('mass_mailing_id', '=', self.id),
|
||||||
|
('state', 'in', ('enqueued', 'sending')),
|
||||||
|
])
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def send_mail(self):
|
||||||
|
if not self.env.context.get('sending_queue_enabled', False):
|
||||||
|
return super(MailMassMailing, self).send_mail()
|
||||||
|
for mailing in self:
|
||||||
|
m_sending = self.env['mail.mass_mailing.sending']
|
||||||
|
sendings = mailing._sendings_get()
|
||||||
|
if sendings:
|
||||||
|
raise UserError(_(
|
||||||
|
"There is another sending task running. "
|
||||||
|
"Please, be patient. You can see all the sending tasks in "
|
||||||
|
"'Sending tasks' tab"
|
||||||
|
))
|
||||||
|
res_ids = mailing.get_recipients(mailing)
|
||||||
|
batch_size = m_sending.batch_size_get()
|
||||||
|
if not res_ids:
|
||||||
|
raise UserError(_("Please select recipients."))
|
||||||
|
sending = m_sending.create({
|
||||||
|
'state': 'draft',
|
||||||
|
'mass_mailing_id': mailing.id,
|
||||||
|
})
|
||||||
|
sending_state = 'enqueued'
|
||||||
|
if len(res_ids) < (batch_size / 2):
|
||||||
|
mailing.with_context(
|
||||||
|
mass_mailing_sending_id=sending.id,
|
||||||
|
sending_queue_enabled=False).send_mail()
|
||||||
|
sending_state = 'sending'
|
||||||
|
sending.state = sending_state
|
||||||
|
mailing.write({
|
||||||
|
'sent_date': fields.Datetime.now(),
|
||||||
|
'state': 'sending',
|
||||||
|
})
|
||||||
|
return True
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def get_recipients(self, mailing):
|
||||||
|
sending = False
|
||||||
|
sending_id = self.env.context.get('mass_mailing_sending_id', False)
|
||||||
|
if sending_id:
|
||||||
|
sending = self.env['mail.mass_mailing.sending'].browse(sending_id)
|
||||||
|
try:
|
||||||
|
res_ids = super(MailMassMailing, self).get_recipients(mailing)
|
||||||
|
except UserError as e:
|
||||||
|
if sending:
|
||||||
|
sending._send_error(e)
|
||||||
|
else:
|
||||||
|
raise
|
||||||
|
return []
|
||||||
|
if sending:
|
||||||
|
res_ids = sending.get_recipient_batch(res_ids)
|
||||||
|
return res_ids
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _compute_pending_count(self):
|
||||||
|
for mailing in self:
|
||||||
|
sendings = mailing._sendings_get()
|
||||||
|
mailing.pending_count = (
|
||||||
|
sum(sendings.mapped('pending_count')) +
|
||||||
|
sum(sendings.mapped('sending_count')))
|
|
@ -0,0 +1,190 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from openerp import api, fields, models, tools
|
||||||
|
|
||||||
|
_logger = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
BATCH_SIZE_DEFAULT = 500
|
||||||
|
|
||||||
|
|
||||||
|
class MailMassMailingSending(models.Model):
|
||||||
|
_name = 'mail.mass_mailing.sending'
|
||||||
|
|
||||||
|
state = fields.Selection([
|
||||||
|
('draft', "Draft"),
|
||||||
|
('enqueued', "Enqueued"),
|
||||||
|
('sending', "Sending"),
|
||||||
|
('sent', "Sent"),
|
||||||
|
('error', "Error"),
|
||||||
|
], string="State", required=True, copy=False, default='enqueued')
|
||||||
|
mass_mailing_id = fields.Many2one(
|
||||||
|
string="Mass mailing", comodel_name='mail.mass_mailing',
|
||||||
|
readonly=True, ondelete='cascade')
|
||||||
|
pending_count = fields.Integer(
|
||||||
|
string="Pending recipients", compute='_compute_pending_count')
|
||||||
|
sending_count = fields.Integer(
|
||||||
|
string="Mails to be sent", compute='_compute_sending_count')
|
||||||
|
sent_count = fields.Integer(
|
||||||
|
string="Sent mails", compute='_compute_sent_count')
|
||||||
|
failed_count = fields.Integer(
|
||||||
|
string="Failed mails", compute='_compute_failed_count')
|
||||||
|
error = fields.Char(string="Error message")
|
||||||
|
date_start = fields.Datetime(
|
||||||
|
string="Date start", default=fields.Datetime.now())
|
||||||
|
date_end = fields.Datetime(string="Date end")
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def batch_size_get(self):
|
||||||
|
m_param = self.env['ir.config_parameter']
|
||||||
|
batch_size = BATCH_SIZE_DEFAULT
|
||||||
|
batch_size_str = m_param.get_param(
|
||||||
|
'mail.mass_mailing.sending.batch_size')
|
||||||
|
if batch_size_str and batch_size_str.isdigit():
|
||||||
|
batch_size = int(batch_size_str)
|
||||||
|
return batch_size
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def pending_emails(self):
|
||||||
|
return self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_sending_id', 'in', self.ids),
|
||||||
|
('scheduled', '!=', False),
|
||||||
|
('sent', '=', False),
|
||||||
|
('exception', '=', False),
|
||||||
|
])
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def get_recipient_batch(self, res_ids):
|
||||||
|
batch_size = self.batch_size_get()
|
||||||
|
already_enqueued = self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_sending_id', 'in', self.ids),
|
||||||
|
])
|
||||||
|
set_ids = set(res_ids)
|
||||||
|
new_ids = list(
|
||||||
|
set_ids - set(already_enqueued.mapped('res_id')))
|
||||||
|
if not self.env.context.get('sending_avoid_batch', False):
|
||||||
|
new_ids = new_ids[:batch_size]
|
||||||
|
if set(new_ids) != set_ids:
|
||||||
|
return new_ids
|
||||||
|
return res_ids
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def pending_recipients(self):
|
||||||
|
self.ensure_one()
|
||||||
|
m_mailing = self.env['mail.mass_mailing'].with_context(
|
||||||
|
mass_mailing_sending_id=self.id, sending_avoid_batch=True)
|
||||||
|
return m_mailing.get_recipients(self.mass_mailing_id)
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def send_mail(self):
|
||||||
|
for sending in self:
|
||||||
|
try:
|
||||||
|
sending.with_context(mass_mailing_sending_id=sending.id).\
|
||||||
|
mass_mailing_id.send_mail()
|
||||||
|
except Exception as e:
|
||||||
|
sending._send_error(e)
|
||||||
|
return True
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _send_error(self, exception):
|
||||||
|
self.ensure_one()
|
||||||
|
self.write({
|
||||||
|
'error': tools.ustr(exception),
|
||||||
|
'state': 'error',
|
||||||
|
'date_end': fields.Datetime.now(),
|
||||||
|
})
|
||||||
|
self.mass_mailing_id.state = 'done'
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _process_enqueued(self):
|
||||||
|
# Create mail_mail objects not created
|
||||||
|
self.ensure_one()
|
||||||
|
if self.pending_recipients():
|
||||||
|
self.send_mail()
|
||||||
|
# If there is no more recipient left, mark as sending
|
||||||
|
if not self.pending_recipients():
|
||||||
|
self.state = 'sending'
|
||||||
|
self._process_sending()
|
||||||
|
elif self.mass_mailing_id.state not in {'sending', 'error'}:
|
||||||
|
self.mass_mailing_id.state = 'sending'
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _process_sending(self):
|
||||||
|
# Check if there is any mail_mail object not sent
|
||||||
|
self.ensure_one()
|
||||||
|
if not self.pending_emails():
|
||||||
|
self.mass_mailing_id.state = 'done'
|
||||||
|
self.write({
|
||||||
|
'state': 'sent',
|
||||||
|
'date_end': fields.Datetime.now(),
|
||||||
|
})
|
||||||
|
elif self.mass_mailing_id.state not in {'sending', 'error'}:
|
||||||
|
self.mass_mailing_id.state = 'sending'
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _process(self):
|
||||||
|
self.ensure_one()
|
||||||
|
method = getattr(self, '_process_%s' % self.state, None)
|
||||||
|
if method and hasattr(method, '__call__'):
|
||||||
|
return method()
|
||||||
|
return False # pragma: no cover
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def sendings_running(self):
|
||||||
|
return self.search([
|
||||||
|
('state', 'in', ('enqueued', 'sending')),
|
||||||
|
])
|
||||||
|
|
||||||
|
@api.model
|
||||||
|
def cron(self):
|
||||||
|
# Process all mail.mass_mailing.sending in enqueue or sending state
|
||||||
|
sendings = self.sendings_running()
|
||||||
|
for sending in sendings:
|
||||||
|
_logger.info("Sending [%d] mass mailing [%d] '%s' (%s)",
|
||||||
|
sending.id, sending.mass_mailing_id.id,
|
||||||
|
sending.mass_mailing_id.name, sending.state)
|
||||||
|
# Process sending using user who created it
|
||||||
|
sending = sending.sudo(user=sending.create_uid.id)
|
||||||
|
ctx = sending.create_uid.context_get()
|
||||||
|
sending.with_context(**ctx)._process()
|
||||||
|
return True
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _compute_pending_count(self):
|
||||||
|
for sending in self.filtered(lambda r: r.state == 'enqueued'):
|
||||||
|
sending.pending_count = len(sending.pending_recipients())
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _compute_sending_count(self):
|
||||||
|
m_stats = self.env['mail.mail.statistics']
|
||||||
|
for sending in self.filtered(
|
||||||
|
lambda r: r.state in {'enqueued', 'sending'}):
|
||||||
|
sending.sending_count = m_stats.search_count([
|
||||||
|
('mass_mailing_sending_id', '=', sending.id),
|
||||||
|
('scheduled', '!=', False),
|
||||||
|
('sent', '=', False),
|
||||||
|
('exception', '=', False),
|
||||||
|
])
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _compute_sent_count(self):
|
||||||
|
m_stats = self.env['mail.mail.statistics']
|
||||||
|
for sending in self:
|
||||||
|
sending.sent_count = m_stats.search_count([
|
||||||
|
('mass_mailing_sending_id', '=', sending.id),
|
||||||
|
('scheduled', '!=', False),
|
||||||
|
('sent', '!=', False),
|
||||||
|
('exception', '=', False),
|
||||||
|
])
|
||||||
|
|
||||||
|
@api.multi
|
||||||
|
def _compute_failed_count(self):
|
||||||
|
m_stats = self.env['mail.mail.statistics']
|
||||||
|
for sending in self:
|
||||||
|
sending.failed_count = m_stats.search_count([
|
||||||
|
('mass_mailing_sending_id', '=', sending.id),
|
||||||
|
('scheduled', '!=', False),
|
||||||
|
('exception', '!=', False),
|
||||||
|
])
|
|
@ -0,0 +1,3 @@
|
||||||
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
|
||||||
|
access_mail_mass_mailing_sending_admin,mail_mass_mailing_sending group_marketing_user,model_mail_mass_mailing_sending,base.group_user,1,1,1,0
|
||||||
|
manage_mail_mass_mailing_sending_admin,mail_mass_mailing_sending group_system,model_mail_mass_mailing_sending,base.group_system,1,1,1,1
|
|
Binary file not shown.
After Width: | Height: | Size: 9.2 KiB |
|
@ -0,0 +1,5 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from . import test_mass_mailing_sending
|
|
@ -0,0 +1,179 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl).
|
||||||
|
|
||||||
|
from openerp.tests.common import TransactionCase
|
||||||
|
from openerp.exceptions import Warning as UserError
|
||||||
|
|
||||||
|
|
||||||
|
class TestMassMailingSending(TransactionCase):
|
||||||
|
def setUp(self, *args, **kwargs):
|
||||||
|
super(TestMassMailingSending, self).setUp(*args, **kwargs)
|
||||||
|
|
||||||
|
self.list = self.env['mail.mass_mailing.list'].create({
|
||||||
|
'name': 'Test list',
|
||||||
|
})
|
||||||
|
# Define a lower batch size for testing purposes
|
||||||
|
self.env['ir.config_parameter'].set_param(
|
||||||
|
'mail.mass_mailing.sending.batch_size', 5)
|
||||||
|
self.contact_a = self.env['mail.mass_mailing.contact'].create({
|
||||||
|
'list_id': self.list.id,
|
||||||
|
'name': 'Test contact A',
|
||||||
|
'email': 'contact_a@example.org',
|
||||||
|
})
|
||||||
|
self.contact_b = self.env['mail.mass_mailing.contact'].create({
|
||||||
|
'list_id': self.list.id,
|
||||||
|
'name': 'Test contact B',
|
||||||
|
'email': 'contact_b@example.org',
|
||||||
|
})
|
||||||
|
for i in range(1, 6):
|
||||||
|
self.env['mail.mass_mailing.contact'].create({
|
||||||
|
'list_id': self.list.id,
|
||||||
|
'name': 'Test contact %s' % i,
|
||||||
|
'email': 'contact_%s@example.org' % i,
|
||||||
|
})
|
||||||
|
self.mass_mailing = self.env['mail.mass_mailing'].create({
|
||||||
|
'name': 'Test mass mailing',
|
||||||
|
'email_from': 'from@example.org',
|
||||||
|
'mailing_model': 'mail.mass_mailing.contact',
|
||||||
|
'mailing_domain': [
|
||||||
|
('list_id', 'in', [self.list.id]),
|
||||||
|
('opt_out', '=', False),
|
||||||
|
],
|
||||||
|
'contact_list_ids': [(6, False, [self.list.id])],
|
||||||
|
'body_html': '<p>Hello world!</p>',
|
||||||
|
'reply_to_mode': 'email',
|
||||||
|
})
|
||||||
|
self.partner = self.env['res.partner'].create({
|
||||||
|
'name': 'Test partner',
|
||||||
|
'email': 'partner@example.org',
|
||||||
|
})
|
||||||
|
self.mass_mailing_short = self.env['mail.mass_mailing'].create({
|
||||||
|
'name': 'Test mass mailing short',
|
||||||
|
'email_from': 'from@example.org',
|
||||||
|
'mailing_model': 'res.partner',
|
||||||
|
'mailing_domain': [
|
||||||
|
('id', 'in', [self.partner.id]),
|
||||||
|
('opt_out', '=', False),
|
||||||
|
],
|
||||||
|
'body_html': '<p>Hello partner!</p>',
|
||||||
|
'reply_to_mode': 'email',
|
||||||
|
})
|
||||||
|
|
||||||
|
def test_cron_contacts(self):
|
||||||
|
self.mass_mailing.with_context(sending_queue_enabled=True).send_mail()
|
||||||
|
sendings = self.env['mail.mass_mailing.sending'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing.id),
|
||||||
|
])
|
||||||
|
stats = self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing.id),
|
||||||
|
])
|
||||||
|
# Sending in 'enqueued' state and 0 email stats created
|
||||||
|
self.assertEqual(1, len(sendings))
|
||||||
|
self.assertEqual(0, len(stats))
|
||||||
|
sending = sendings[0]
|
||||||
|
self.assertEqual('enqueued', sending.state)
|
||||||
|
self.assertEqual(7, sending.pending_count)
|
||||||
|
self.assertEqual('sending', self.mass_mailing.state)
|
||||||
|
self.assertEqual(7, self.mass_mailing.pending_count)
|
||||||
|
# Create email stats
|
||||||
|
sending.cron()
|
||||||
|
stats = self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing.id),
|
||||||
|
])
|
||||||
|
self.env.invalidate_all()
|
||||||
|
# Sending in 'enqueued' state and 5 stats created, 2 pending, 5 sending
|
||||||
|
self.assertEqual(5, len(stats))
|
||||||
|
self.assertEqual('enqueued', sending.state)
|
||||||
|
self.assertEqual(2, sending.pending_count)
|
||||||
|
self.assertEqual(5, sending.sending_count)
|
||||||
|
self.assertEqual('sending', self.mass_mailing.state)
|
||||||
|
for stat in stats:
|
||||||
|
if stat.mail_mail_id:
|
||||||
|
stat.mail_mail_id.send()
|
||||||
|
self.env.invalidate_all()
|
||||||
|
# Check that 5 emails are already sent
|
||||||
|
self.assertEqual(0, sending.sending_count)
|
||||||
|
self.assertEqual(5, sending.sent_count)
|
||||||
|
sending.cron()
|
||||||
|
stats = self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing.id),
|
||||||
|
])
|
||||||
|
self.env.invalidate_all()
|
||||||
|
# Sending in 'sending' state and 7 stats created, 0 pending, 2 sending
|
||||||
|
self.assertEqual(7, len(stats))
|
||||||
|
self.assertEqual('sending', sending.state)
|
||||||
|
self.assertEqual(0, sending.pending_count)
|
||||||
|
self.assertEqual(2, sending.sending_count)
|
||||||
|
self.assertEqual('sending', self.mass_mailing.state)
|
||||||
|
for stat in stats:
|
||||||
|
if stat.mail_mail_id:
|
||||||
|
stat.mail_mail_id.send()
|
||||||
|
self.env.invalidate_all()
|
||||||
|
# Check that 7 emails are already sent
|
||||||
|
self.assertEqual('sent', sending.state)
|
||||||
|
self.assertEqual(0, sending.sending_count)
|
||||||
|
self.assertEqual(7, sending.sent_count)
|
||||||
|
self.assertEqual(0, sending.failed_count)
|
||||||
|
self.assertEqual('done', self.mass_mailing.state)
|
||||||
|
|
||||||
|
def test_cron_partners(self):
|
||||||
|
self.mass_mailing_short.with_context(
|
||||||
|
sending_queue_enabled=True).send_mail()
|
||||||
|
sendings = self.env['mail.mass_mailing.sending'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing_short.id),
|
||||||
|
])
|
||||||
|
stats = self.env['mail.mail.statistics'].search([
|
||||||
|
('mass_mailing_id', '=', self.mass_mailing_short.id),
|
||||||
|
])
|
||||||
|
# Sending in 'draft' state and 1 email stats created
|
||||||
|
self.assertEqual(1, len(sendings))
|
||||||
|
self.assertEqual(1, len(stats))
|
||||||
|
sending = sendings[0]
|
||||||
|
self.assertEqual('sending', sending.state)
|
||||||
|
self.assertEqual(0, sending.pending_count)
|
||||||
|
self.assertEqual('sending', self.mass_mailing_short.state)
|
||||||
|
self.assertEqual(1, self.mass_mailing_short.pending_count)
|
||||||
|
for stat in stats:
|
||||||
|
if stat.mail_mail_id:
|
||||||
|
stat.mail_mail_id.send()
|
||||||
|
self.env.invalidate_all()
|
||||||
|
# Check that 1 email are already sent
|
||||||
|
self.assertEqual('sent', sending.state)
|
||||||
|
self.assertEqual(0, sending.sending_count)
|
||||||
|
self.assertEqual(1, sending.sent_count)
|
||||||
|
self.assertEqual(0, sending.failed_count)
|
||||||
|
self.assertEqual('done', self.mass_mailing_short.state)
|
||||||
|
|
||||||
|
def test_concurrent(self):
|
||||||
|
self.mass_mailing.with_context(sending_queue_enabled=True).send_mail()
|
||||||
|
with self.assertRaises(UserError):
|
||||||
|
self.mass_mailing.with_context(
|
||||||
|
sending_queue_enabled=True).send_mail()
|
||||||
|
|
||||||
|
def test_read_group(self):
|
||||||
|
groups = self.env['mail.mass_mailing'].read_group(
|
||||||
|
[('sent_date', '<', '1900-12-31')], ['state', 'name'], ['state'])
|
||||||
|
self.assertTrue([
|
||||||
|
x for x in groups if (
|
||||||
|
x['state_count'] == 0 and x['state'][0] == 'sending')
|
||||||
|
])
|
||||||
|
|
||||||
|
def test_no_recipients(self):
|
||||||
|
empty_list = self.env['mail.mass_mailing.list'].create({
|
||||||
|
'name': 'Test list with no recipients',
|
||||||
|
})
|
||||||
|
mass_mailing = self.env['mail.mass_mailing'].create({
|
||||||
|
'name': 'Test mass mailing with no recipients',
|
||||||
|
'email_from': 'from@example.org',
|
||||||
|
'mailing_model': 'mail.mass_mailing.contact',
|
||||||
|
'mailing_domain': [
|
||||||
|
('list_id', 'in', [empty_list.id]),
|
||||||
|
('opt_out', '=', False),
|
||||||
|
],
|
||||||
|
'contact_list_ids': [(6, False, [empty_list.id])],
|
||||||
|
'body_html': '<p>Hello no one!</p>',
|
||||||
|
'reply_to_mode': 'email',
|
||||||
|
})
|
||||||
|
with self.assertRaises(UserError):
|
||||||
|
mass_mailing.send_mail()
|
|
@ -0,0 +1,81 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||||
|
|
||||||
|
<openerp>
|
||||||
|
<data>
|
||||||
|
|
||||||
|
<record model="ir.ui.view" id="view_mail_mass_mailing_sending_tree">
|
||||||
|
<field name="name">mail.mass_mailing.sending.tree</field>
|
||||||
|
<field name="model">mail.mass_mailing.sending</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<tree string="Mass mailing sendings">
|
||||||
|
<field name="mass_mailing_id"/>
|
||||||
|
<field name="date_start"/>
|
||||||
|
<field name="state"/>
|
||||||
|
<field name="pending_count"/>
|
||||||
|
<field name="sending_count"/>
|
||||||
|
<field name="sent_count"/>
|
||||||
|
<field name="failed_count"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record model="ir.ui.view" id="view_mail_mass_mailing_sending_form">
|
||||||
|
<field name="name">mail.mass_mailing.sending.form</field>
|
||||||
|
<field name="model">mail.mass_mailing.sending</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<form string="Mass mailing sending">
|
||||||
|
<group>
|
||||||
|
<group>
|
||||||
|
<field name="mass_mailing_id"/>
|
||||||
|
<field name="state"/>
|
||||||
|
<field name="date_start"/>
|
||||||
|
<field name="date_end"/>
|
||||||
|
</group>
|
||||||
|
<group>
|
||||||
|
<field name="pending_count"/>
|
||||||
|
<field name="sending_count"/>
|
||||||
|
<field name="sent_count"/>
|
||||||
|
<field name="failed_count"/>
|
||||||
|
</group>
|
||||||
|
</group>
|
||||||
|
<group attrs="{'invisible': [('error', '=', False)]}">
|
||||||
|
<field name="error"/>
|
||||||
|
</group>
|
||||||
|
</form>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record model="ir.ui.view" id="view_mail_mass_mailing_sending_search">
|
||||||
|
<field name="name">mail.mass_mailing.sending.search</field>
|
||||||
|
<field name="model">mail.mass_mailing.sending</field>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<search string="Mass mailing sending">
|
||||||
|
<field name="mass_mailing_id"/>
|
||||||
|
<filter string="Enqueued" name="enqueued" domain="[('state', '=', 'enqueued')]"/>
|
||||||
|
<filter string="Sending" name="sending" domain="[('state', '=', 'sending')]"/>
|
||||||
|
<filter string="Running" name="running" domain="[('state', '!=', 'sent')]"/>
|
||||||
|
<filter string="Sent" name="sent" domain="[('state', '=' ,'sent')]"/>
|
||||||
|
<group expand="0" string="Group By">
|
||||||
|
<filter string="Mass mailing" name="group_mass_mailing" context="{'group_by': 'mass_mailing_id'}"/>
|
||||||
|
</group>
|
||||||
|
</search>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<record id="action_view_mail_mass_mailing_sending" model="ir.actions.act_window">
|
||||||
|
<field name="name">Mass mailing sendings</field>
|
||||||
|
<field name="res_model">mail.mass_mailing.sending</field>
|
||||||
|
<field name="view_type">form</field>
|
||||||
|
<field name="view_mode">tree,form</field>
|
||||||
|
<field name="context">{'search_default_running': 1}</field>
|
||||||
|
<field name="search_view_id" ref="view_mail_mass_mailing_sending_search"/>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
<menuitem name="Mass mailing sending" id="menu_mass_mailing_sending"
|
||||||
|
parent="base.menu_email" sequence="60"
|
||||||
|
action="action_view_mail_mass_mailing_sending"/>
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</openerp>
|
|
@ -0,0 +1,48 @@
|
||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright 2016 Antonio Espinosa <antonio.espinosa@tecnativa.com>
|
||||||
|
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
|
||||||
|
|
||||||
|
<openerp>
|
||||||
|
<data>
|
||||||
|
|
||||||
|
<record model="ir.ui.view" id="view_mail_mass_mailing_form">
|
||||||
|
<field name="name">Add pending emails to be sent</field>
|
||||||
|
<field name="model">mail.mass_mailing</field>
|
||||||
|
<field name="inherit_id" ref="mass_mailing.view_mail_mass_mailing_form"/>
|
||||||
|
<field name="arch" type="xml">
|
||||||
|
<xpath expr="//button[@name='send_mail'][1]" position="attributes">
|
||||||
|
<attribute name="attrs">{'invisible': ['|', ('state', 'in', ('done', 'sending')), ('body_html', '=', False)]}</attribute>
|
||||||
|
<attribute name="context">{'sending_queue_enabled': True}</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="//button[@name='send_mail'][2]" position="attributes">
|
||||||
|
<attribute name="attrs">{'invisible': ['|', ('state', 'not in', ('done', )), ('body_html', '=', False)]}</attribute>
|
||||||
|
<attribute name="context">{'sending_queue_enabled': True}</attribute>
|
||||||
|
</xpath>
|
||||||
|
<xpath expr="div[@class='oe_form_box_info oe_text_center']" position="attributes">
|
||||||
|
<attribute name="attrs">{'invisible': [('pending_count', '=', 0)]}</attribute>
|
||||||
|
</xpath>
|
||||||
|
<field name="scheduled" position="attributes">
|
||||||
|
<attribute name="invisible">1</attribute>
|
||||||
|
</field>
|
||||||
|
<field name="scheduled" position="after">
|
||||||
|
<field name="pending_count" class="oe_inline"/>
|
||||||
|
</field>
|
||||||
|
<notebook position="inside">
|
||||||
|
<page string="Sending tasks">
|
||||||
|
<field name="mass_mailing_sending_ids" nolabel="1">
|
||||||
|
<tree string="Sending tasks">
|
||||||
|
<field name="date_start"/>
|
||||||
|
<field name="state"/>
|
||||||
|
<field name="pending_count"/>
|
||||||
|
<field name="sending_count"/>
|
||||||
|
<field name="sent_count"/>
|
||||||
|
<field name="failed_count"/>
|
||||||
|
</tree>
|
||||||
|
</field>
|
||||||
|
</page>
|
||||||
|
</notebook>
|
||||||
|
</field>
|
||||||
|
</record>
|
||||||
|
|
||||||
|
</data>
|
||||||
|
</openerp>
|
Loading…
Reference in New Issue