Jairo Llopis
6 years ago
committed by
fkantelberg
29 changed files with 2069 additions and 0 deletions
-
1privacy_consent/README.rst
-
3privacy_consent/__init__.py
-
29privacy_consent/__manifest__.py
-
1privacy_consent/controllers/__init__.py
-
47privacy_consent/controllers/main.py
-
23privacy_consent/data/ir_actions_server.xml
-
16privacy_consent/data/ir_cron.xml
-
155privacy_consent/data/mail.xml
-
663privacy_consent/i18n/es.po
-
5privacy_consent/models/__init__.py
-
54privacy_consent/models/mail_mail.py
-
33privacy_consent/models/mail_template.py
-
144privacy_consent/models/privacy_activity.py
-
201privacy_consent/models/privacy_consent.py
-
32privacy_consent/models/res_partner.py
-
3privacy_consent/readme/CONTRIBUTORS.rst
-
7privacy_consent/readme/DESCRIPTION.rst
-
15privacy_consent/readme/INSTALL.rst
-
69privacy_consent/readme/USAGE.rst
-
3privacy_consent/security/ir.model.access.csv
-
BINprivacy_consent/static/description/icon.png
-
63privacy_consent/templates/form.xml
-
1privacy_consent/tests/__init__.py
-
247privacy_consent/tests/test_consent.py
-
89privacy_consent/views/privacy_activity.xml
-
113privacy_consent/views/privacy_consent.xml
-
35privacy_consent/views/res_partner.xml
-
1privacy_consent/wizards/__init__.py
-
16privacy_consent/wizards/mail_compose_message.py
@ -0,0 +1 @@ |
|||||
|
|
@ -0,0 +1,3 @@ |
|||||
|
from . import controllers |
||||
|
from . import models |
||||
|
from . import wizards |
@ -0,0 +1,29 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
{ |
||||
|
"name": "Privacy - Consent", |
||||
|
"summary": "Allow people to explicitly accept or reject inclusion " |
||||
|
"in some activity, GDPR compliant", |
||||
|
"version": "10.0.1.0.0", |
||||
|
"development_status": "Production/Stable", |
||||
|
"category": "Privacy", |
||||
|
"website": "https://github.com/OCA/management-activity", |
||||
|
"author": "Tecnativa, Odoo Community Association (OCA)", |
||||
|
"license": "AGPL-3", |
||||
|
"application": False, |
||||
|
"installable": True, |
||||
|
"depends": [ |
||||
|
"privacy", |
||||
|
], |
||||
|
"data": [ |
||||
|
"security/ir.model.access.csv", |
||||
|
"data/ir_actions_server.xml", |
||||
|
"data/ir_cron.xml", |
||||
|
"data/mail.xml", |
||||
|
"templates/form.xml", |
||||
|
"views/privacy_consent.xml", |
||||
|
"views/privacy_activity.xml", |
||||
|
"views/res_partner.xml", |
||||
|
], |
||||
|
} |
@ -0,0 +1 @@ |
|||||
|
from . import main |
@ -0,0 +1,47 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from datetime import datetime |
||||
|
|
||||
|
from werkzeug.exceptions import NotFound |
||||
|
|
||||
|
from odoo.http import Controller, request, route |
||||
|
|
||||
|
from odoo.addons.web.controllers.main import ensure_db |
||||
|
|
||||
|
|
||||
|
class ConsentController(Controller): |
||||
|
@route("/privacy/consent/<any(accept,reject):choice>/" |
||||
|
"<int:consent_id>/<token>", |
||||
|
type="http", auth="none", website=True) |
||||
|
def consent(self, choice, consent_id, token, *args, **kwargs): |
||||
|
"""Process user's consent acceptance or rejection.""" |
||||
|
ensure_db() |
||||
|
try: |
||||
|
# If there's a website, we need a user to render the template |
||||
|
request.uid = request.website.user_id.id |
||||
|
except AttributeError: |
||||
|
# If there's no website, the default is OK |
||||
|
pass |
||||
|
consent = request.env["privacy.consent"] \ |
||||
|
.with_context(subject_answering=True) \ |
||||
|
.sudo().browse(consent_id) |
||||
|
if not (consent.exists() and consent._token() == token): |
||||
|
raise NotFound |
||||
|
if consent.partner_id.lang: |
||||
|
consent = consent.with_context(lang=consent.partner_id.lang) |
||||
|
request.context = consent.env.context |
||||
|
consent.action_answer(choice == "accept", self._metadata()) |
||||
|
return request.render("privacy_consent.form", { |
||||
|
"consent": consent, |
||||
|
}) |
||||
|
|
||||
|
def _metadata(self): |
||||
|
return (u"User agent: {}\n" |
||||
|
u"Remote IP: {}\n" |
||||
|
u"Date and time: {:%Y-%m-%d %H:%M:%S}").format( |
||||
|
request.httprequest.environ.get("HTTP_USER_AGENT"), |
||||
|
request.httprequest.environ.get("REMOTE_ADDRESS"), |
||||
|
datetime.now(), |
||||
|
) |
@ -0,0 +1,23 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<record id="update_opt_out" model="ir.actions.server"> |
||||
|
<field name="name">Update partner's opt out</field> |
||||
|
<field name="model_id" ref="model_privacy_consent"/> |
||||
|
<field name="crud_model_id" ref="base.model_res_partner"/> |
||||
|
<field name="state">object_write</field> |
||||
|
<field name="use_write">expression</field> |
||||
|
<field name="write_expression">record.partner_id</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="update_opt_out_line_1" model="ir.server.object.lines"> |
||||
|
<field name="server_id" ref="update_opt_out"/> |
||||
|
<field name="col1" ref="mail.field_res_partner_opt_out"/> |
||||
|
<field name="type">equation</field> |
||||
|
<field name="value">not record.accepted</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
@ -0,0 +1,16 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<record id="cron_auto_consent" model="ir.cron"> |
||||
|
<field name="name">Request automatic data processing consents</field> |
||||
|
<field name="model">privacy.activity</field> |
||||
|
<field name="function">_cron_new_consents</field> |
||||
|
<field name="interval_number">1</field> |
||||
|
<field name="interval_type">work_days</field> |
||||
|
<field name="numbercall">-1</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
@ -0,0 +1,155 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<!-- Mail templates --> |
||||
|
<record id="template_consent" model="mail.template"> |
||||
|
<field name="name">Personal data processing consent request</field> |
||||
|
<field name="subject">Data processing consent request for ${object.activity_id.display_name|safe}</field> |
||||
|
<field name="model_id" ref="model_privacy_consent"/> |
||||
|
<field name="use_default_to" eval="True"/> |
||||
|
<field name="body_html" type="xml"> |
||||
|
<div style="background:#F3F5F6;color:#515166;padding:25px 0px;font-family:Arial,Helvetica,sans-serif;font-size:14px;"> |
||||
|
<table style="width:600px;margin:5px auto;"> |
||||
|
<tbody> |
||||
|
<tr> |
||||
|
<td> |
||||
|
<a href="/"> |
||||
|
<img src="/logo" alt="${object.activity_id.controller_id.display_name|safe}" style="vertical-align:baseline;max-width:100px;"/> |
||||
|
</a> |
||||
|
</td> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
<table style="width:600px;margin:0px auto;background:white;border:1px solid #e1e1e1;"> |
||||
|
<tbody> |
||||
|
<tr> |
||||
|
<td colspan="2" style="padding:15px 20px 0px 20px; font-size:16px;"> |
||||
|
<p> |
||||
|
Hello, ${object.partner_id.name|safe} |
||||
|
</p> |
||||
|
<p> |
||||
|
We contacted you to ask you to give us your explicit consent to include your data in a data processing activity called |
||||
|
<b>${object.activity_id.display_name|safe}</b>, property of |
||||
|
<i>${object.activity_id.controller_id.display_name|safe}</i> |
||||
|
</p> |
||||
|
${object.description or ""} |
||||
|
<p> |
||||
|
% if object.state == "answered": |
||||
|
The last time you answered, you |
||||
|
% elif object.state == "sent": |
||||
|
If you do nothing, we will assume you have |
||||
|
% endif |
||||
|
|
||||
|
% if object.accepted: |
||||
|
<b>accepted</b> |
||||
|
% else: |
||||
|
<b>rejected</b> |
||||
|
% endif |
||||
|
such data processing. |
||||
|
</p> |
||||
|
<p> |
||||
|
You can update your preferences below: |
||||
|
</p> |
||||
|
</td> |
||||
|
</tr> |
||||
|
<tr> |
||||
|
<td style="padding:15px 20px 0px 20px; font-size:16px; text-align:right;"> |
||||
|
<a href="/privacy/consent/accept/" style="background-color: #449d44; padding: 12px; font-weight: 12px; text-decoration: none; color: #fff; border-radius: 5px; font-size:16px;"> |
||||
|
Accept |
||||
|
</a> |
||||
|
</td> |
||||
|
<td style="padding:15px 20px 0px 20px; font-size:16px; text-align:left;"> |
||||
|
<a href="/privacy/consent/reject/" style="background-color: #d9534f; padding: 12px; font-weight: 12px; text-decoration: none; color: #fff; border-radius: 5px; font-size:16px;"> |
||||
|
Reject |
||||
|
</a> |
||||
|
</td> |
||||
|
</tr> |
||||
|
<tr> |
||||
|
<td colspan="2" style="padding:15px 20px 15px 20px; font-size:16px;"> |
||||
|
<p> |
||||
|
If you need further information, please respond to this email and we will attend your request as soon as possible. |
||||
|
</p> |
||||
|
<p> |
||||
|
Thank you! |
||||
|
</p> |
||||
|
</td> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
<table style="width:600px;margin:0px auto;text-align:center;"> |
||||
|
<tbody> |
||||
|
<tr> |
||||
|
<td style="padding-top:10px;font-size: 12px;"> |
||||
|
<p> |
||||
|
Sent by |
||||
|
<a href="/" style="color:#717188;">${object.activity_id.controller_id.display_name|safe}</a>. |
||||
|
</p> |
||||
|
</td> |
||||
|
</tr> |
||||
|
</tbody> |
||||
|
</table> |
||||
|
</div> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<!-- Mail subtypes --> |
||||
|
<record id="mt_consent_consent_new" model="mail.message.subtype"> |
||||
|
<field name="name">New Consent</field> |
||||
|
<field name="description">Privacy consent request created</field> |
||||
|
<field name="res_model">privacy.consent</field> |
||||
|
<field name="default" eval="False"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
</record> |
||||
|
<record id="mt_consent_acceptance_changed" model="mail.message.subtype"> |
||||
|
<field name="name">Acceptance Changed by Subject</field> |
||||
|
<field name="description">Acceptance status updated by subject</field> |
||||
|
<field name="res_model">privacy.consent</field> |
||||
|
<field name="default" eval="False"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
</record> |
||||
|
<record id="mt_consent_state_changed" model="mail.message.subtype"> |
||||
|
<field name="name">State Changed</field> |
||||
|
<field name="description">Privacy consent request state changed</field> |
||||
|
<field name="res_model">privacy.consent</field> |
||||
|
<field name="default" eval="False"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
</record> |
||||
|
|
||||
|
<record id="mt_activity_consent_new" model="mail.message.subtype"> |
||||
|
<field name="name">New Consent</field> |
||||
|
<field name="description">Privacy consent request created</field> |
||||
|
<field name="res_model">privacy.activity</field> |
||||
|
<field name="default" eval="True"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
<field name="parent_id" ref="mt_consent_consent_new"/> |
||||
|
<field name="relation_field">activity_id</field> |
||||
|
</record> |
||||
|
<record id="mt_activity_acceptance_changed" model="mail.message.subtype"> |
||||
|
<field name="name">Acceptance Changed</field> |
||||
|
<field name="description">Privacy consent request acceptance status changed</field> |
||||
|
<field name="res_model">privacy.activity</field> |
||||
|
<field name="default" eval="True"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
<field name="parent_id" ref="mt_consent_acceptance_changed"/> |
||||
|
<field name="relation_field">activity_id</field> |
||||
|
</record> |
||||
|
<record id="mt_activity_state_changed" model="mail.message.subtype"> |
||||
|
<field name="name">State Changed</field> |
||||
|
<field name="description">Privacy consent request state changed</field> |
||||
|
<field name="res_model">privacy.activity</field> |
||||
|
<field name="default" eval="False"/> |
||||
|
<field name="hidden" eval="False"/> |
||||
|
<field name="internal" eval="True"/> |
||||
|
<field name="parent_id" ref="mt_consent_state_changed"/> |
||||
|
<field name="relation_field">activity_id</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
@ -0,0 +1,663 @@ |
|||||
|
# Translation of Odoo Server. |
||||
|
# This file contains the translation of the following modules: |
||||
|
# * privacy_consent |
||||
|
# |
||||
|
msgid "" |
||||
|
msgstr "" |
||||
|
"Project-Id-Version: Odoo Server 10.0\n" |
||||
|
"Report-Msgid-Bugs-To: \n" |
||||
|
"POT-Creation-Date: 2018-07-11 08:38+0000\n" |
||||
|
"PO-Revision-Date: 2018-07-11 11:07+0200\n" |
||||
|
"Language-Team: \n" |
||||
|
"MIME-Version: 1.0\n" |
||||
|
"Content-Type: text/plain; charset=UTF-8\n" |
||||
|
"Content-Transfer-Encoding: 8bit\n" |
||||
|
"Plural-Forms: nplurals=2; plural=(n != 1);\n" |
||||
|
"X-Generator: Poedit 2.0.8\n" |
||||
|
"Last-Translator: Jairo Llopis <yajo.sk8@gmail.com>\n" |
||||
|
"Language: es_ES\n" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.template,body_html:privacy_consent.template_consent |
||||
|
msgid "" |
||||
|
"<?xml version=\"1.0\"?>\n" |
||||
|
"<div style=\"background:#F3F5F6;color:#515166;padding:25px 0px;font-family:" |
||||
|
"Arial,Helvetica,sans-serif;font-size:14px;\">\n" |
||||
|
" <table style=\"width:600px;margin:5px auto;\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td>\n" |
||||
|
" <a href=\"/\">\n" |
||||
|
" <img src=\"/logo\" alt=\"${object." |
||||
|
"activity_id.controller_id.display_name|safe}\" style=\"vertical-align:" |
||||
|
"baseline;max-width:100px;\"/>\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" <table style=\"width:600px;margin:0px auto;background:white;" |
||||
|
"border:1px solid #e1e1e1;\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td colspan=\"2\" style=\"padding:15px 20px 0px " |
||||
|
"20px; font-size:16px;\">\n" |
||||
|
" <p>\n" |
||||
|
" Hello, ${object.partner_id.name|safe}\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" We contacted you to ask you to give us " |
||||
|
"your explicit consent to include your data in a data processing activity " |
||||
|
"called\n" |
||||
|
" <b>${object.activity_id.display_name|" |
||||
|
"safe}</b>, property of\n" |
||||
|
" <i>${object.activity_id.controller_id." |
||||
|
"display_name|safe}</i>\n" |
||||
|
" </p>\n" |
||||
|
" ${object.description or \"\"}\n" |
||||
|
" <p>\n" |
||||
|
" % if object.state == \"answered\":\n" |
||||
|
" The last time you answered, you\n" |
||||
|
" % elif object.state == \"sent\":\n" |
||||
|
" If you do nothing, we will assume " |
||||
|
"you have\n" |
||||
|
" % endif\n" |
||||
|
"\n" |
||||
|
" % if object.accepted:\n" |
||||
|
" <b>accepted</b>\n" |
||||
|
" % else:\n" |
||||
|
" <b>rejected</b>\n" |
||||
|
" % endif\n" |
||||
|
" such data processing.\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" You can update your preferences below:\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" <tr>\n" |
||||
|
" <td style=\"padding:15px 20px 0px 20px; font-" |
||||
|
"size:16px; text-align:right;\">\n" |
||||
|
" <a href=\"/privacy/consent/accept/\" style=" |
||||
|
"\"background-color: #449d44; padding: 12px; font-weight: 12px; text-" |
||||
|
"decoration: none; color: #fff; border-radius: 5px; font-size:16px;\">\n" |
||||
|
" Accept\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" <td style=\"padding:15px 20px 0px 20px; font-" |
||||
|
"size:16px; text-align:left;\">\n" |
||||
|
" <a href=\"/privacy/consent/reject/\" style=" |
||||
|
"\"background-color: #d9534f; padding: 12px; font-weight: 12px; text-" |
||||
|
"decoration: none; color: #fff; border-radius: 5px; font-size:16px;\">\n" |
||||
|
" Reject\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" <tr>\n" |
||||
|
" <td colspan=\"2\" style=\"padding:15px 20px 15px " |
||||
|
"20px; font-size:16px;\">\n" |
||||
|
" <p>\n" |
||||
|
" If you need further information, please " |
||||
|
"respond to this email and we will attend your request as soon as possible.\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" Thank you!\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" <table style=\"width:600px;margin:0px auto;text-align:center;" |
||||
|
"\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td style=\"padding-top:10px;font-size: 12px;" |
||||
|
"\">\n" |
||||
|
" <p>\n" |
||||
|
" Sent by\n" |
||||
|
" <a href=\"/\" style=\"color:#717188;\">" |
||||
|
"${object.activity_id.controller_id.display_name|safe}</a>.\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" </div>\n" |
||||
|
" " |
||||
|
msgstr "" |
||||
|
"<?xml version=\"1.0\"?>\n" |
||||
|
"<div style=\"background:#F3F5F6;color:#515166;padding:25px 0px;font-family:" |
||||
|
"Arial,Helvetica,sans-serif;font-size:14px;\">\n" |
||||
|
" <table style=\"width:600px;margin:5px auto;\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td>\n" |
||||
|
" <a href=\"/\">\n" |
||||
|
" <img src=\"/logo\" alt=\"${object." |
||||
|
"activity_id.controller_id.display_name|safe}\" style=\"vertical-align:" |
||||
|
"baseline;max-width:100px;\"/>\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" <table style=\"width:600px;margin:0px auto;background:white;" |
||||
|
"border:1px solid #e1e1e1;\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td colspan=\"2\" style=\"padding:15px 20px 0px " |
||||
|
"20px; font-size:16px;\">\n" |
||||
|
" <p>\n" |
||||
|
" Hola, ${object.partner_id.name|safe}\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" Le hemos contactado para pedirle su " |
||||
|
"consentimiento explícito para incluir sus datos en una actividad de " |
||||
|
"tratamiento llamada\n" |
||||
|
" <b>${object.activity_id.display_name|" |
||||
|
"safe}</b>, propiedad de\n" |
||||
|
" <i>${object.activity_id.controller_id." |
||||
|
"display_name|safe}</i>\n" |
||||
|
" </p>\n" |
||||
|
" ${object.description or \"\"}\n" |
||||
|
" <p>\n" |
||||
|
" % if object.state == \"answered\":\n" |
||||
|
" Según su última respuesta,\n" |
||||
|
" % elif object.state == \"sent\":\n" |
||||
|
" Si no recibimos respuesta, " |
||||
|
"asumiremos que\n" |
||||
|
" % endif\n" |
||||
|
"\n" |
||||
|
" % if object.accepted:\n" |
||||
|
" <b>ha aceptado</b>\n" |
||||
|
" % else:\n" |
||||
|
" <b>ha rechazado</b>\n" |
||||
|
" % endif\n" |
||||
|
" dicho procesamiento de datos.\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" Puede cambiar sus preferencias aquí " |
||||
|
"abajo:\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" <tr>\n" |
||||
|
" <td style=\"padding:15px 20px 0px 20px; font-" |
||||
|
"size:16px; text-align:right;\">\n" |
||||
|
" <a href=\"/privacy/consent/accept/\" style=" |
||||
|
"\"background-color: #449d44; padding: 12px; font-weight: 12px; text-" |
||||
|
"decoration: none; color: #fff; border-radius: 5px; font-size:16px;\">\n" |
||||
|
" Aceptar\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" <td style=\"padding:15px 20px 0px 20px; font-" |
||||
|
"size:16px; text-align:left;\">\n" |
||||
|
" <a href=\"/privacy/consent/reject/\" style=" |
||||
|
"\"background-color: #d9534f; padding: 12px; font-weight: 12px; text-" |
||||
|
"decoration: none; color: #fff; border-radius: 5px; font-size:16px;\">\n" |
||||
|
" Rechazar\n" |
||||
|
" </a>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" <tr>\n" |
||||
|
" <td colspan=\"2\" style=\"padding:15px 20px 15px " |
||||
|
"20px; font-size:16px;\">\n" |
||||
|
" <p>\n" |
||||
|
" Si necesita más información, por favor " |
||||
|
"responda a este correo electrónico y atenderemos su solicitud a la mayor " |
||||
|
"brevedad posible.\n" |
||||
|
" </p>\n" |
||||
|
" <p>\n" |
||||
|
" ¡Gracias!\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" <table style=\"width:600px;margin:0px auto;text-align:center;" |
||||
|
"\">\n" |
||||
|
" <tbody>\n" |
||||
|
" <tr>\n" |
||||
|
" <td style=\"padding-top:10px;font-size: 12px;" |
||||
|
"\">\n" |
||||
|
" <p>\n" |
||||
|
" Enviado por\n" |
||||
|
" <a href=\"/\" style=\"color:#717188;\">" |
||||
|
"${object.activity_id.controller_id.display_name|safe}</a>.\n" |
||||
|
" </p>\n" |
||||
|
" </td>\n" |
||||
|
" </tr>\n" |
||||
|
" </tbody>\n" |
||||
|
" </table>\n" |
||||
|
" </div>\n" |
||||
|
" " |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_activity_acceptance_changed |
||||
|
msgid "Acceptance Changed" |
||||
|
msgstr "Aceptación cambiada" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_consent_acceptance_changed |
||||
|
msgid "Acceptance Changed by Subject" |
||||
|
msgstr "Aceptación cambiada por el interesado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_consent_acceptance_changed |
||||
|
msgid "Acceptance status updated by subject" |
||||
|
msgstr "Estado de aceptación modificado por el interesado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_accepted |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_search |
||||
|
msgid "Accepted" |
||||
|
msgstr "Aceptado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_default_consent |
||||
|
msgid "Accepted by default" |
||||
|
msgstr "Aceptado por defecto" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_active |
||||
|
msgid "Active" |
||||
|
msgstr "Activo" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_activity_id |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_search |
||||
|
msgid "Activity" |
||||
|
msgstr "Actividad" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: selection:privacy.consent,state:0 |
||||
|
msgid "Answered" |
||||
|
msgstr "Respondido" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_search |
||||
|
msgid "Archived" |
||||
|
msgstr "Archivado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_form |
||||
|
msgid "Ask for consent" |
||||
|
msgstr "Solicitar consentimiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_required |
||||
|
msgid "Ask subjects for consent" |
||||
|
msgstr "Solicitar consentimiento a los interesados" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: selection:privacy.activity,consent_required:0 |
||||
|
msgid "Automatically" |
||||
|
msgstr "Automáticamente" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: selection:privacy.consent,state:0 |
||||
|
msgid "Awaiting response" |
||||
|
msgstr "Esperando respuesta" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.activity_form |
||||
|
msgid "Consent" |
||||
|
msgstr "Consentimiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_privacy_consent |
||||
|
msgid "Consent of data processing" |
||||
|
msgstr "Consentimiento para tratamiento de datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_template_default_body_html |
||||
|
msgid "Consent template default body html" |
||||
|
msgstr "HTML por defecto para el cuerpo de la plantilla de consentimiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_template_default_subject |
||||
|
msgid "Consent template default subject" |
||||
|
msgstr "HTML por defecto para el asunto de la plantilla de consentimiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.actions.act_window,name:privacy_consent.consent_action |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_count |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_ids |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_res_partner_privacy_consent_count |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_res_users_privacy_consent_count |
||||
|
#: model:ir.ui.menu,name:privacy_consent.menu_privacy_consent |
||||
|
msgid "Consents" |
||||
|
msgstr "Consents" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_create_uid |
||||
|
msgid "Created by" |
||||
|
msgstr "Creado por" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_create_date |
||||
|
msgid "Created on" |
||||
|
msgstr "Creado el" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_privacy_activity |
||||
|
msgid "Data processing activities" |
||||
|
msgstr "Actividades de tratamiento de datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.template,subject:privacy_consent.template_consent |
||||
|
msgid "" |
||||
|
"Data processing consent request for ${object.activity_id.display_name|safe}" |
||||
|
msgstr "" |
||||
|
"Solicitud de consentimiento para el tratamiento de datos personales para " |
||||
|
"${object.activity_id.display_name|safe}" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_display_name |
||||
|
msgid "Display Name" |
||||
|
msgstr "Nombre a mostrar" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: selection:privacy.consent,state:0 |
||||
|
msgid "Draft" |
||||
|
msgstr "Borrador" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: sql_constraint:privacy.consent:0 |
||||
|
msgid "Duplicated partner in this data processing activity" |
||||
|
msgstr "Contacto duplicado en esta actividad de tratamiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_mail_template |
||||
|
msgid "Email Templates" |
||||
|
msgstr "Plantillas de correo electrónico" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_mail_compose_message |
||||
|
msgid "Email composition wizard" |
||||
|
msgstr "Asistente de redacción de correo electrónico" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_consent_template_id |
||||
|
msgid "Email template" |
||||
|
msgstr "Plantilla de correo electrónico" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_activity_consent_template_id |
||||
|
msgid "" |
||||
|
"Email to be sent to subjects to ask for consent. A good template should " |
||||
|
"include details about the current consent request status, how to change it, " |
||||
|
"and where to get more information." |
||||
|
msgstr "" |
||||
|
"Correo electrónico a enviar a los interesados para solicitarles el " |
||||
|
"consentimiento. Una buena plantilla debería incluir detalles sobre el estado " |
||||
|
"actual del consentimiento, cómo cambiarlo, y dónde obtener más información." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_activity_consent_required |
||||
|
msgid "" |
||||
|
"Enable if you need to track any kind of consent from the affected subjects" |
||||
|
msgstr "" |
||||
|
"Actívelo si necesita registrar cualquier tipo de consentimiento de los " |
||||
|
"interesados." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.activity_form |
||||
|
msgid "Generate and send missing consent requests" |
||||
|
msgstr "Generar y enviar solicitudes de consentimiento faltantes" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.activity_form |
||||
|
msgid "Generate missing draft consent requests" |
||||
|
msgstr "Generar borradores de las solicitudes de consentimiento faltantes" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: code:addons/privacy_consent/models/privacy_activity.py:140 |
||||
|
#, python-format |
||||
|
msgid "Generated consents" |
||||
|
msgstr "Consentimientos generados" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_search |
||||
|
msgid "Group By" |
||||
|
msgstr "Agrupar por" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "Hello," |
||||
|
msgstr "Hola," |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "I <b>accept</b> this processing of my data" |
||||
|
msgstr "<b>Acepto</b> este tratamiento de mis datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "I <b>reject</b> this processing of my data" |
||||
|
msgstr "<b>Rechazo</b> este tratamiento de mis datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_id |
||||
|
msgid "ID" |
||||
|
msgstr "ID" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "If it was a mistake, you can undo it here:" |
||||
|
msgstr "Si ha sido un error, puede deshacerlo aquí:" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_consent_accepted |
||||
|
msgid "" |
||||
|
"Indicates current acceptance status, which can come from subject's last " |
||||
|
"answer, or from the default specified in the related data processing " |
||||
|
"activity." |
||||
|
msgstr "" |
||||
|
"Indica el estado actual de la aceptación, el cual puede venir de la última " |
||||
|
"respuesta del interesado, o del estado por defecto especificado en la " |
||||
|
"actividad de tratamiento relacionada." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent___last_update |
||||
|
msgid "Last Modified on" |
||||
|
msgstr "Última modificación en" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_write_uid |
||||
|
msgid "Last Updated by" |
||||
|
msgstr "Última actualización por" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_write_date |
||||
|
msgid "Last Updated on" |
||||
|
msgstr "Última actualización el" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_last_metadata |
||||
|
msgid "Last metadata" |
||||
|
msgstr "Últimos metadatos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: selection:privacy.activity,consent_required:0 |
||||
|
msgid "Manually" |
||||
|
msgstr "Manualmente" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_consent_last_metadata |
||||
|
msgid "Metadata from the last acceptance or rejection by the subject" |
||||
|
msgstr "" |
||||
|
"Metadatos de la última aceptación o denegación por parte del interesado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: code:addons/privacy_consent/models/mail_template.py:25 |
||||
|
#, python-format |
||||
|
msgid "" |
||||
|
"Missing privacy consent link placeholders. You need at least these two " |
||||
|
"links:\n" |
||||
|
"<a href=\"%s\">Accept</a>\n" |
||||
|
"<a href=\"%s\">Reject</a>" |
||||
|
msgstr "" |
||||
|
"Faltan los marcadores de posición de los enlaces para el consentimiento. " |
||||
|
"Necesita al menos estos dos enlaces:\n" |
||||
|
"<a href=\"%s\">Aceptar</a>\n" |
||||
|
"<a href=\"%s\">Rechazar</a>" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_activity_consent_new |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_consent_consent_new |
||||
|
msgid "New Consent" |
||||
|
msgstr "Nuevo consentimiento" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_mail_mail |
||||
|
msgid "Outgoing Mails" |
||||
|
msgstr "Correos electrónicos salientes" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model,name:privacy_consent.model_res_partner |
||||
|
msgid "Partner" |
||||
|
msgstr "Contacto" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_activity_acceptance_changed |
||||
|
msgid "Privacy consent request acceptance status changed" |
||||
|
msgstr "" |
||||
|
"El estado de aceptación de la solicitud de consentimiento para el " |
||||
|
"tratamiento de datos ha cambiado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_activity_consent_new |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_consent_consent_new |
||||
|
msgid "Privacy consent request created" |
||||
|
msgstr "" |
||||
|
"La solicitud de consentimiento para el tratamiento de datos ha sido creada" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_activity_state_changed |
||||
|
#: model:mail.message.subtype,description:privacy_consent.mt_consent_state_changed |
||||
|
msgid "Privacy consent request state changed" |
||||
|
msgstr "" |
||||
|
"El estado de la solicitud de consentimiento para el tratamiento de datos ha " |
||||
|
"cambiado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_res_partner_privacy_consent_count |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_res_users_privacy_consent_count |
||||
|
msgid "Privacy consent requests amount" |
||||
|
msgstr "Cantidad de solicitudes de consentimiento para el tratamiento de datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_res_partner_privacy_consent_ids |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_res_users_privacy_consent_ids |
||||
|
msgid "Privacy consents" |
||||
|
msgstr "Consentimientos para el tratamiento de datos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: code:addons/privacy_consent/models/privacy_activity.py:100 |
||||
|
#, python-format |
||||
|
msgid "Require consent is available only for subjects in current database." |
||||
|
msgstr "" |
||||
|
"La opción de exigir consentimiento solo está disponible para interesados que " |
||||
|
"se encuentren en esta misma base de datos." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_activity_server_action_id |
||||
|
msgid "" |
||||
|
"Run this action when a new consent request is created or its acceptance " |
||||
|
"status is updated." |
||||
|
msgstr "" |
||||
|
"Ejecutar esta acción cuando se cree una nueva solicitud de consentimiento, o " |
||||
|
"cuando su estado de aceptación cambie." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_activity_server_action_id |
||||
|
msgid "Server action" |
||||
|
msgstr "Acción de servidor" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_activity_default_consent |
||||
|
msgid "Should we assume the subject has accepted if we receive no response?" |
||||
|
msgstr "" |
||||
|
"¿Hay que asumir que el interesado ha aceptado si no recibimos respuesta?" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "Sincerely,<br/>" |
||||
|
msgstr "Atentamente,<br/>" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: code:addons/privacy_consent/models/privacy_activity.py:92 |
||||
|
#, python-format |
||||
|
msgid "Specify a mail template to ask automated consent." |
||||
|
msgstr "" |
||||
|
"Especifique una plantilla de correo electrónico para solicitar " |
||||
|
"automáticamente el consentimiento." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_state |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.consent_search |
||||
|
msgid "State" |
||||
|
msgstr "Estado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_activity_state_changed |
||||
|
#: model:mail.message.subtype,name:privacy_consent.mt_consent_state_changed |
||||
|
msgid "State Changed" |
||||
|
msgstr "El estado ha cambiado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,field_description:privacy_consent.field_privacy_consent_partner_id |
||||
|
msgid "Subject" |
||||
|
msgstr "Interesado" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.model.fields,help:privacy_consent.field_privacy_consent_partner_id |
||||
|
msgid "Subject asked for consent." |
||||
|
msgstr "Interesado a quien se le pide el consentimiento." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "Thank you!" |
||||
|
msgstr "¡Gracias!" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "Thanks for your response." |
||||
|
msgstr "Gracias por su respuesta." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.activity_form |
||||
|
msgid "This could send many consent emails, are you sure to proceed?" |
||||
|
msgstr "" |
||||
|
"Esto podría enviar muchos correos electrónicos solicitando consentimiento " |
||||
|
"para el tratamiento de datos, ¿seguro que quiere continuar?" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.actions.server,name:privacy_consent.update_opt_out |
||||
|
msgid "Update partner's opt out" |
||||
|
msgstr "Sincronizar la opción del contacto para recibir o no envíos masivos" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "" |
||||
|
"We asked you to authorize us to process your data in this data processing " |
||||
|
"activity:" |
||||
|
msgstr "" |
||||
|
"Le hemos solicitado que nos autorice para procesar sus datos personales en " |
||||
|
"esta actividad de tratamiento:" |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "We have recorded this action on your side." |
||||
|
msgstr "Hemos registrado esta acción por su parte." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "You have <b class=\"text-danger\">rejected</b> such processing." |
||||
|
msgstr "Ha <b class=\"text-danger\">rechazado</b> dicho tratamiento." |
||||
|
|
||||
|
#. module: privacy_consent |
||||
|
#: model:ir.ui.view,arch_db:privacy_consent.form |
||||
|
msgid "You have <b class=\"text-success\">accepted</b> such processing." |
||||
|
msgstr "Ha <b class=\"text-success\">aceptado</b> dicho tratamiento." |
@ -0,0 +1,5 @@ |
|||||
|
from . import mail_mail |
||||
|
from . import mail_template |
||||
|
from . import privacy_activity |
||||
|
from . import privacy_consent |
||||
|
from . import res_partner |
@ -0,0 +1,54 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import models |
||||
|
|
||||
|
|
||||
|
class MailMail(models.Model): |
||||
|
_inherit = "mail.mail" |
||||
|
|
||||
|
def _postprocess_sent_message(self, mail_sent=True): |
||||
|
"""Write consent status after sending message.""" |
||||
|
if mail_sent and self.env.context.get('mark_consent_sent'): |
||||
|
# Get all mails sent to consents |
||||
|
consent_mails = self.filtered( |
||||
|
lambda one: one.mail_message_id.model == "privacy.consent" |
||||
|
) |
||||
|
# Get related draft consents |
||||
|
consents = self.env["privacy.consent"].browse( |
||||
|
consent_mails.mapped("mail_message_id.res_id"), |
||||
|
self._prefetch |
||||
|
).filtered(lambda one: one.state == "draft") |
||||
|
# Set as sent |
||||
|
consents.write({ |
||||
|
"state": "sent", |
||||
|
}) |
||||
|
return super(MailMail, self)._postprocess_sent_message(mail_sent) |
||||
|
|
||||
|
def send_get_mail_body(self, partner=None): |
||||
|
"""Replace privacy consent magic links. |
||||
|
|
||||
|
This replacement is done here instead of directly writing it into |
||||
|
the ``mail.template`` to avoid writing the tokeinzed URL |
||||
|
in the mail thread for the ``privacy.consent`` record, |
||||
|
which would enable any reader of such thread to impersonate the |
||||
|
subject and choose in its behalf. |
||||
|
""" |
||||
|
result = super(MailMail, self).send_get_mail_body(partner=partner) |
||||
|
# Avoid polluting other model mails |
||||
|
if self.env.context.get("active_model") != "privacy.consent": |
||||
|
return result |
||||
|
# Tokenize consent links |
||||
|
consent = self.env["privacy.consent"] \ |
||||
|
.browse(self.mail_message_id.res_id) \ |
||||
|
.with_prefetch(self._prefetch) |
||||
|
result = result.replace( |
||||
|
"/privacy/consent/accept/", |
||||
|
consent._url(True), |
||||
|
) |
||||
|
result = result.replace( |
||||
|
"/privacy/consent/reject/", |
||||
|
consent._url(False), |
||||
|
) |
||||
|
return result |
@ -0,0 +1,33 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from lxml import html |
||||
|
|
||||
|
from odoo import _, api, models |
||||
|
from odoo.exceptions import ValidationError |
||||
|
|
||||
|
|
||||
|
class MailTemplate(models.Model): |
||||
|
_inherit = "mail.template" |
||||
|
|
||||
|
@api.constrains("body_html", "model") |
||||
|
def _check_consent_links_in_body_html(self): |
||||
|
"""Body for ``privacy.consent`` templates needs placeholder links.""" |
||||
|
links = [u"//a[@href='/privacy/consent/{}/']".format(action) |
||||
|
for action in ("accept", "reject")] |
||||
|
for one in self: |
||||
|
if one.model != "privacy.consent": |
||||
|
continue |
||||
|
doc = html.document_fromstring(one.body_html) |
||||
|
for link in links: |
||||
|
if not doc.xpath(link): |
||||
|
raise ValidationError(_( |
||||
|
"Missing privacy consent link placeholders. " |
||||
|
"You need at least these two links:\n" |
||||
|
'<a href="%s">Accept</a>\n' |
||||
|
'<a href="%s">Reject</a>' |
||||
|
) % ( |
||||
|
"/privacy/consent/accept/", |
||||
|
"/privacy/consent/reject/", |
||||
|
)) |
@ -0,0 +1,144 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import _, api, fields, models |
||||
|
from odoo.exceptions import ValidationError |
||||
|
from odoo.tools.safe_eval import safe_eval |
||||
|
|
||||
|
|
||||
|
class PrivacyActivity(models.Model): |
||||
|
_inherit = 'privacy.activity' |
||||
|
|
||||
|
server_action_id = fields.Many2one( |
||||
|
"ir.actions.server", |
||||
|
"Server action", |
||||
|
domain=[ |
||||
|
("model_id.model", "=", "privacy.consent"), |
||||
|
], |
||||
|
help="Run this action when a new consent request is created or its " |
||||
|
"acceptance status is updated.", |
||||
|
) |
||||
|
consent_ids = fields.One2many( |
||||
|
"privacy.consent", |
||||
|
"activity_id", |
||||
|
"Consents", |
||||
|
) |
||||
|
consent_count = fields.Integer( |
||||
|
"Consents", |
||||
|
compute="_compute_consent_count", |
||||
|
) |
||||
|
consent_required = fields.Selection( |
||||
|
[("auto", "Automatically"), ("manual", "Manually")], |
||||
|
"Ask subjects for consent", |
||||
|
help="Enable if you need to track any kind of consent " |
||||
|
"from the affected subjects", |
||||
|
) |
||||
|
consent_template_id = fields.Many2one( |
||||
|
"mail.template", |
||||
|
"Email template", |
||||
|
default=lambda self: self._default_consent_template_id(), |
||||
|
domain=[ |
||||
|
("model", "=", "privacy.consent"), |
||||
|
], |
||||
|
help="Email to be sent to subjects to ask for consent. " |
||||
|
"A good template should include details about the current " |
||||
|
"consent request status, how to change it, and where to " |
||||
|
"get more information.", |
||||
|
) |
||||
|
default_consent = fields.Boolean( |
||||
|
"Accepted by default", |
||||
|
help="Should we assume the subject has accepted if we receive no " |
||||
|
"response?", |
||||
|
) |
||||
|
|
||||
|
# Hidden helpers help user design new templates |
||||
|
consent_template_default_body_html = fields.Text( |
||||
|
compute="_compute_consent_template_defaults", |
||||
|
) |
||||
|
consent_template_default_subject = fields.Char( |
||||
|
compute="_compute_consent_template_defaults", |
||||
|
) |
||||
|
|
||||
|
@api.model |
||||
|
def _default_consent_template_id(self): |
||||
|
return self.env.ref("privacy_consent.template_consent", False) |
||||
|
|
||||
|
@api.depends("consent_ids") |
||||
|
def _compute_consent_count(self): |
||||
|
groups = self.env["privacy.consent"].read_group( |
||||
|
[("activity_id", "in", self.ids)], |
||||
|
["activity_id"], |
||||
|
["activity_id"], |
||||
|
) |
||||
|
for group in groups: |
||||
|
self.browse(group["activity_id"][0], self._prefetch) \ |
||||
|
.consent_count = group["activity_id_count"] |
||||
|
|
||||
|
def _compute_consent_template_defaults(self): |
||||
|
"""Used in context values, to help users design new templates.""" |
||||
|
template = self._default_consent_template_id() |
||||
|
if template: |
||||
|
self.update({ |
||||
|
"consent_template_default_body_html": template.body_html, |
||||
|
"consent_template_default_subject": template.subject, |
||||
|
}) |
||||
|
|
||||
|
@api.constrains("consent_required", "consent_template_id") |
||||
|
def _check_auto_consent_has_template(self): |
||||
|
"""Require a mail template to automate consent requests.""" |
||||
|
for one in self: |
||||
|
if one.consent_required == "auto" and not one.consent_template_id: |
||||
|
raise ValidationError(_( |
||||
|
"Specify a mail template to ask automated consent." |
||||
|
)) |
||||
|
|
||||
|
@api.constrains("consent_required", "subject_find") |
||||
|
def _check_consent_required_subject_find(self): |
||||
|
for one in self: |
||||
|
if one.consent_required and not one.subject_find: |
||||
|
raise ValidationError(_( |
||||
|
"Require consent is available only for subjects " |
||||
|
"in current database." |
||||
|
)) |
||||
|
|
||||
|
@api.model |
||||
|
def _cron_new_consents(self): |
||||
|
"""Ask all missing automatic consent requests.""" |
||||
|
automatic = self.search([("consent_required", "=", "auto")]) |
||||
|
automatic.action_new_consents() |
||||
|
|
||||
|
@api.onchange("consent_required") |
||||
|
def _onchange_consent_required_subject_find(self): |
||||
|
"""Find subjects automatically if we require their consent.""" |
||||
|
if self.consent_required: |
||||
|
self.subject_find = True |
||||
|
|
||||
|
def action_new_consents(self): |
||||
|
"""Generate new consent requests.""" |
||||
|
consents = self.env["privacy.consent"] |
||||
|
# Skip activitys where consent is not required |
||||
|
for one in self.with_context(active_test=False) \ |
||||
|
.filtered("consent_required"): |
||||
|
domain = safe_eval(one.subject_domain) |
||||
|
domain += [ |
||||
|
("id", "not in", one.mapped("consent_ids.partner_id").ids), |
||||
|
("email", "!=", False), |
||||
|
] |
||||
|
# Create missing consent requests |
||||
|
for missing in self.env["res.partner"].search(domain): |
||||
|
consents |= consents.create({ |
||||
|
"partner_id": missing.id, |
||||
|
"accepted": one.default_consent, |
||||
|
"activity_id": one.id, |
||||
|
}) |
||||
|
# Send consent request emails for automatic activitys |
||||
|
consents.action_auto_ask() |
||||
|
# Redirect user to new consent requests generated |
||||
|
return { |
||||
|
"domain": [("id", "in", consents.ids)], |
||||
|
"name": _("Generated consents"), |
||||
|
"res_model": consents._name, |
||||
|
"type": "ir.actions.act_window", |
||||
|
"view_mode": "tree,form", |
||||
|
} |
@ -0,0 +1,201 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
import hashlib |
||||
|
import hmac |
||||
|
|
||||
|
from odoo import api, fields, models |
||||
|
|
||||
|
|
||||
|
class PrivacyConsent(models.Model): |
||||
|
_name = 'privacy.consent' |
||||
|
_description = "Consent of data processing" |
||||
|
_inherit = "mail.thread" |
||||
|
_rec_name = "partner_id" |
||||
|
_sql_constraints = [ |
||||
|
("unique_partner_activity", "UNIQUE(partner_id, activity_id)", |
||||
|
"Duplicated partner in this data processing activity"), |
||||
|
] |
||||
|
|
||||
|
active = fields.Boolean( |
||||
|
default=True, |
||||
|
index=True, |
||||
|
) |
||||
|
accepted = fields.Boolean( |
||||
|
track_visibility="onchange", |
||||
|
help="Indicates current acceptance status, which can come from " |
||||
|
"subject's last answer, or from the default specified in the " |
||||
|
"related data processing activity.", |
||||
|
) |
||||
|
last_metadata = fields.Text( |
||||
|
readonly=True, |
||||
|
track_visibility="onchange", |
||||
|
help="Metadata from the last acceptance or rejection by the subject", |
||||
|
) |
||||
|
partner_id = fields.Many2one( |
||||
|
"res.partner", |
||||
|
"Subject", |
||||
|
required=True, |
||||
|
readonly=True, |
||||
|
track_visibility="onchange", |
||||
|
help="Subject asked for consent.", |
||||
|
) |
||||
|
activity_id = fields.Many2one( |
||||
|
"privacy.activity", |
||||
|
"Activity", |
||||
|
readonly=True, |
||||
|
required=True, |
||||
|
track_visibility="onchange", |
||||
|
) |
||||
|
state = fields.Selection( |
||||
|
selection=[ |
||||
|
("draft", "Draft"), |
||||
|
("sent", "Awaiting response"), |
||||
|
("answered", "Answered"), |
||||
|
], |
||||
|
default="draft", |
||||
|
readonly=True, |
||||
|
required=True, |
||||
|
track_visibility="onchange", |
||||
|
) |
||||
|
|
||||
|
def _track_subtype(self, init_values): |
||||
|
"""Return specific subtypes.""" |
||||
|
if self.env.context.get("subject_answering"): |
||||
|
return "privacy_consent.mt_consent_acceptance_changed" |
||||
|
if "activity_id" in init_values or "partner_id" in init_values: |
||||
|
return "privacy_consent.mt_consent_consent_new" |
||||
|
if "state" in init_values: |
||||
|
return "privacy_consent.mt_consent_state_changed" |
||||
|
return super(PrivacyConsent, self)._track_subtype(init_values) |
||||
|
|
||||
|
def _token(self): |
||||
|
"""Secret token to publicly authenticate this record.""" |
||||
|
secret = self.env["ir.config_parameter"].sudo().get_param( |
||||
|
"database.secret") |
||||
|
params = "{}-{}-{}-{}".format( |
||||
|
self.env.cr.dbname, |
||||
|
self.id, |
||||
|
self.partner_id.id, |
||||
|
self.activity_id.id, |
||||
|
) |
||||
|
return hmac.new( |
||||
|
secret.encode('utf-8'), |
||||
|
params.encode('utf-8'), |
||||
|
hashlib.sha512, |
||||
|
).hexdigest() |
||||
|
|
||||
|
def _url(self, accept): |
||||
|
"""Tokenized URL to let subject decide consent. |
||||
|
|
||||
|
:param bool accept: |
||||
|
Indicates if you want the acceptance URL, or the rejection one. |
||||
|
""" |
||||
|
return "/privacy/consent/{}/{}/{}?db={}".format( |
||||
|
"accept" if accept else "reject", |
||||
|
self.id, |
||||
|
self._token(), |
||||
|
self.env.cr.dbname, |
||||
|
) |
||||
|
|
||||
|
def _send_consent_notification(self): |
||||
|
"""Send email notification to subject.""" |
||||
|
consents_by_template = {} |
||||
|
for one in self.with_context(tpl_force_default_to=True, |
||||
|
mail_notify_user_signature=False, |
||||
|
mail_auto_subscribe_no_notify=True, |
||||
|
mark_consent_sent=True): |
||||
|
# Group consents by template, to send in batch where possible |
||||
|
template_id = one.activity_id.consent_template_id.id |
||||
|
consents_by_template.setdefault(template_id, one) |
||||
|
consents_by_template[template_id] |= one |
||||
|
# Send emails |
||||
|
for template_id, consents in consents_by_template.items(): |
||||
|
consents.message_post_with_template( |
||||
|
template_id, |
||||
|
# This mode always sends email, regardless of partner's |
||||
|
# notification preferences; we use it here because it's very |
||||
|
# likely that we are asking authorisation to send emails |
||||
|
composition_mode="mass_mail", |
||||
|
) |
||||
|
|
||||
|
def _run_action(self): |
||||
|
"""Execute server action defined in data processing activity.""" |
||||
|
for one in self: |
||||
|
# Always skip draft consents |
||||
|
if one.state == "draft": |
||||
|
continue |
||||
|
action = one.activity_id.server_action_id.with_context( |
||||
|
active_id=one.id, |
||||
|
active_ids=one.ids, |
||||
|
active_model=one._name, |
||||
|
) |
||||
|
action.run() |
||||
|
|
||||
|
@api.model |
||||
|
def create(self, vals): |
||||
|
"""Run server action on create.""" |
||||
|
result = super(PrivacyConsent, self).create(vals) |
||||
|
# Sync the default acceptance status |
||||
|
result.sudo()._run_action() |
||||
|
return result |
||||
|
|
||||
|
def write(self, vals): |
||||
|
"""Run server action on update.""" |
||||
|
result = super(PrivacyConsent, self).write(vals) |
||||
|
self._run_action() |
||||
|
return result |
||||
|
|
||||
|
def message_get_suggested_recipients(self): |
||||
|
result = super(PrivacyConsent, self) \ |
||||
|
.message_get_suggested_recipients() |
||||
|
reason = self._fields["partner_id"].string |
||||
|
for one in self: |
||||
|
one._message_add_suggested_recipient( |
||||
|
result, |
||||
|
partner=one.partner_id, |
||||
|
reason=reason, |
||||
|
) |
||||
|
return result |
||||
|
|
||||
|
def action_manual_ask(self): |
||||
|
"""Let user manually ask for consent.""" |
||||
|
return { |
||||
|
"context": { |
||||
|
"default_composition_mode": "mass_mail", |
||||
|
"default_model": self._name, |
||||
|
"default_res_id": self.id, |
||||
|
"default_template_id": self.activity_id.consent_template_id.id, |
||||
|
"default_use_template": True, |
||||
|
"mark_consent_sent": True, |
||||
|
"tpl_force_default_to": True, |
||||
|
}, |
||||
|
"force_email": True, |
||||
|
"res_model": "mail.compose.message", |
||||
|
"target": "new", |
||||
|
"type": "ir.actions.act_window", |
||||
|
"view_mode": "form", |
||||
|
} |
||||
|
|
||||
|
def action_auto_ask(self): |
||||
|
"""Automatically ask for consent.""" |
||||
|
templated = self.filtered("activity_id.consent_template_id") |
||||
|
automated = templated.filtered( |
||||
|
lambda one: one.activity_id.consent_required == "auto") |
||||
|
automated._send_consent_notification() |
||||
|
|
||||
|
def action_answer(self, answer, metadata=False): |
||||
|
"""Process answer. |
||||
|
|
||||
|
:param bool answer: |
||||
|
Did the subject accept? |
||||
|
|
||||
|
:param str metadata: |
||||
|
Metadata from last user acceptance or rejection request. |
||||
|
""" |
||||
|
self.write({ |
||||
|
"state": "answered", |
||||
|
"accepted": answer, |
||||
|
"last_metadata": metadata, |
||||
|
}) |
@ -0,0 +1,32 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import api, fields, models |
||||
|
|
||||
|
|
||||
|
class ResPartner(models.Model): |
||||
|
_inherit = "res.partner" |
||||
|
|
||||
|
privacy_consent_ids = fields.One2many( |
||||
|
"privacy.consent", |
||||
|
"partner_id", |
||||
|
"Privacy consents", |
||||
|
) |
||||
|
privacy_consent_count = fields.Integer( |
||||
|
"Consents", |
||||
|
compute="_compute_privacy_consent_count", |
||||
|
help="Privacy consent requests amount", |
||||
|
) |
||||
|
|
||||
|
@api.depends("privacy_consent_ids") |
||||
|
def _compute_privacy_consent_count(self): |
||||
|
"""Count consent requests.""" |
||||
|
groups = self.env["privacy.consent"].read_group( |
||||
|
[("partner_id", "in", self.ids)], |
||||
|
["partner_id"], |
||||
|
["partner_id"], |
||||
|
) |
||||
|
for group in groups: |
||||
|
self.browse(group["partner_id"][0], self._prefetch) \ |
||||
|
.privacy_consent_count = group["partner_id_count"] |
@ -0,0 +1,3 @@ |
|||||
|
* `Tecnativa <https://www.tecnativa.com>`_: |
||||
|
|
||||
|
* Jairo Llopis |
@ -0,0 +1,7 @@ |
|||||
|
This module allows the user to define a set of subjects (partners) |
||||
|
affected by any data processing activity, and establish |
||||
|
a process to ask them for consent to include them in that activity. |
||||
|
|
||||
|
For those that need explicit consent as a lawfulness base for personal data |
||||
|
processing, as required by GDPR (article 6.1.a), this module provides the |
||||
|
needed tools to automate it. |
@ -0,0 +1,15 @@ |
|||||
|
You may want to install, along with this module, one of OCA's |
||||
|
``mail_tracking`` module collection, such as ``mail_tracking_mailgun``, so |
||||
|
you can provide more undeniable proof that some consent request was sent, and |
||||
|
to whom. |
||||
|
|
||||
|
However, the most important proof to provide is the answer itself (more than |
||||
|
the question), and this addon provides enough tooling for that. |
||||
|
|
||||
|
Multi-database instances |
||||
|
~~~~~~~~~~~~~~~~~~~~~~~~ |
||||
|
|
||||
|
To enable multi-database support, you must load this addon as a server-wide |
||||
|
addon. Example command to boot Odoo:: |
||||
|
|
||||
|
odoo-bin --load=web,privacy_consent |
@ -0,0 +1,69 @@ |
|||||
|
New options for data processing activities: |
||||
|
|
||||
|
#. Go to *Privacy > Master Data > Activities* and create one. |
||||
|
|
||||
|
#. Give it a name, such as *Sending mass mailings to customers*. |
||||
|
|
||||
|
#. Go to tab *Consent* and choose one option in *Ask subjects for consent*: |
||||
|
|
||||
|
* *Manual* tells the activity that you will want to create and send the |
||||
|
consent requests manually, and only provides some helpers for you to |
||||
|
be able to batch-generate them. |
||||
|
|
||||
|
* *Automatic* enables this module's full power: send all consent requests |
||||
|
to selected partners automatically, every day and under your demand. |
||||
|
|
||||
|
#. When you do this, all the consent-related options appear. Configure them: |
||||
|
|
||||
|
* A smart button tells you how many consents have been generated, and lets you |
||||
|
access them. |
||||
|
|
||||
|
* Choose one *Email template* to send to subjects. This email itself is what |
||||
|
asks for consent, and it gets recorded, to serve as a proof that it was sent. |
||||
|
The module provides a default template that should be good for most usage |
||||
|
cases; and if you create one directly from that field, some good defaults |
||||
|
are provided for your comfortability. |
||||
|
|
||||
|
* *Subjects filter* defines what partners will be elegible for inclusion in |
||||
|
this data processing activity. |
||||
|
|
||||
|
* You can enable *Accepted by default* if you want to assume subjects |
||||
|
accepted their data processing. You should possibly consult your |
||||
|
lawyer to use this. |
||||
|
|
||||
|
* You can choose a *Server action* (developer mode only) that will |
||||
|
be executed whenever a new non-draft consent request is created, |
||||
|
or when its acceptance status changes. |
||||
|
|
||||
|
This module supplies a server action by default, called |
||||
|
*Update partner's opt out*, that syncs the acceptance status with the |
||||
|
partner's *Elegible for mass mailings* option. |
||||
|
|
||||
|
#. Click on *Generate consent requests* link to create new consent requests. |
||||
|
|
||||
|
* If you chose *Manual* mode, all missing consent request are created as |
||||
|
drafts, and nothing else is done now. |
||||
|
|
||||
|
* If you chose *Automatic* mode, also those requests are sent to subjects |
||||
|
and set as *Sent*. |
||||
|
|
||||
|
#. You will be presented with the list of just-created consent requests. |
||||
|
See below. |
||||
|
|
||||
|
New options for consent requests: |
||||
|
|
||||
|
#. Access the consent requests by either: |
||||
|
|
||||
|
* Generating new consent requests from a data processing activity. |
||||
|
|
||||
|
* Pressing the *Consents* smart button in a data processing activity. |
||||
|
|
||||
|
* Going to *Privacy > Privacy > Consents*. |
||||
|
|
||||
|
#. A consent will include the partner, the activity, the acceptance status, |
||||
|
and the request state. |
||||
|
|
||||
|
#. You can manually ask for consent by pressing the button labeled as |
||||
|
*Ask for consent*. |
||||
|
|
||||
|
#. All consent requests and responses are recorded in the mail thread below. |
@ -0,0 +1,3 @@ |
|||||
|
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink |
||||
|
privacy_consent_read,Permission to read consents,model_privacy_consent,privacy.group_data_protection_user,1,0,0,0 |
||||
|
privacy_consent_write,Permission to write consents,model_privacy_consent,privacy.group_data_protection_manager,1,1,1,1 |
After Width: 128 | Height: 128 | Size: 9.2 KiB |
@ -0,0 +1,63 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<template id="form" name="Consent response processed"> |
||||
|
<!-- Use web.login_layout because it gets automatically wrapped |
||||
|
by website layout if website is installed, and otherwise includes |
||||
|
all possibly needed assets --> |
||||
|
<t t-call="web.login_layout"> |
||||
|
<div class="container readable"> |
||||
|
<div class="jumbotron"> |
||||
|
<h1>Thank you!</h1> |
||||
|
<p> |
||||
|
Hello, <b t-esc="consent.partner_id.display_name"/> |
||||
|
</p> |
||||
|
<p> |
||||
|
We asked you to authorize us to process your data in this data processing activity: |
||||
|
<b t-esc="consent.activity_id.display_name"/> |
||||
|
</p> |
||||
|
<t t-raw="consent.activity_id.description or ''"/> |
||||
|
<p t-if="consent.accepted"> |
||||
|
You have <b class="text-success">accepted</b> such processing. |
||||
|
</p> |
||||
|
<p t-else=""> |
||||
|
You have <b class="text-danger">rejected</b> such processing. |
||||
|
</p> |
||||
|
<p> |
||||
|
We have recorded this action on your side. |
||||
|
</p> |
||||
|
<p> |
||||
|
If it was a mistake, you can undo it here: |
||||
|
<div class="text-center"> |
||||
|
<a |
||||
|
t-if="consent.accepted" |
||||
|
t-att-href="consent._url(False)" |
||||
|
class="btn btn-danger btn-lg" |
||||
|
> |
||||
|
I <b>reject</b> this processing of my data |
||||
|
</a> |
||||
|
<a |
||||
|
t-else="" |
||||
|
t-att-href="consent._url(True)" |
||||
|
class="btn btn-success btn-lg" |
||||
|
> |
||||
|
I <b>accept</b> this processing of my data |
||||
|
</a> |
||||
|
</div> |
||||
|
</p> |
||||
|
<p> |
||||
|
Thanks for your response. |
||||
|
</p> |
||||
|
<p class="text-muted"> |
||||
|
Sincerely,<br/> |
||||
|
<i t-raw="consent.activity_id.controller_id.with_context(show_address=True, html_format=True).name_get()[0][1]"/> |
||||
|
</p> |
||||
|
</div> |
||||
|
</div> |
||||
|
</t> |
||||
|
</template> |
||||
|
|
||||
|
</data> |
@ -0,0 +1 @@ |
|||||
|
from . import test_consent |
@ -0,0 +1,247 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from contextlib import contextmanager |
||||
|
|
||||
|
from odoo.exceptions import ValidationError |
||||
|
from odoo.tests.common import at_install, post_install, HttpCase |
||||
|
|
||||
|
|
||||
|
@at_install(False) |
||||
|
@post_install(True) |
||||
|
class ActivityCase(HttpCase): |
||||
|
def setUp(self): |
||||
|
super(ActivityCase, self).setUp() |
||||
|
# HACK https://github.com/odoo/odoo/issues/12237 |
||||
|
# TODO Remove hack in v12 |
||||
|
self._oldenv = self.env |
||||
|
self.env = self._oldenv(self.cursor()) |
||||
|
# HACK end |
||||
|
self.cron = self.env.ref("privacy_consent.cron_auto_consent") |
||||
|
self.update_opt_out = self.env.ref("privacy_consent.update_opt_out") |
||||
|
self.mt_consent_consent_new = self.env.ref( |
||||
|
"privacy_consent.mt_consent_consent_new") |
||||
|
self.mt_consent_acceptance_changed = self.env.ref( |
||||
|
"privacy_consent.mt_consent_acceptance_changed") |
||||
|
self.mt_consent_state_changed = self.env.ref( |
||||
|
"privacy_consent.mt_consent_state_changed") |
||||
|
# Some partners to ask for consent |
||||
|
self.partners = self.env["res.partner"] |
||||
|
self.partners += self.partners.create({ |
||||
|
"name": "consent-partner-0", |
||||
|
"email": "partner0@example.com", |
||||
|
"notify_email": "none", |
||||
|
"opt_out": False, |
||||
|
}) |
||||
|
self.partners += self.partners.create({ |
||||
|
"name": "consent-partner-1", |
||||
|
"email": "partner1@example.com", |
||||
|
"notify_email": "always", |
||||
|
"opt_out": True, |
||||
|
}) |
||||
|
self.partners += self.partners.create({ |
||||
|
"name": "consent-partner-2", |
||||
|
"email": "partner2@example.com", |
||||
|
"opt_out": False, |
||||
|
}) |
||||
|
# Partner without email, on purpose |
||||
|
self.partners += self.partners.create({ |
||||
|
"name": "consent-partner-3", |
||||
|
"opt_out": True, |
||||
|
}) |
||||
|
# Activity without consent |
||||
|
self.activity_noconsent = self.env["privacy.activity"].create({ |
||||
|
"name": "activity_noconsent", |
||||
|
"description": "I'm activity 1", |
||||
|
}) |
||||
|
# Activity with auto consent, for all partners |
||||
|
self.activity_auto = self.env["privacy.activity"].create({ |
||||
|
"name": "activity_auto", |
||||
|
"description": "I'm activity auto", |
||||
|
"subject_find": True, |
||||
|
"subject_domain": repr([("id", "in", self.partners.ids)]), |
||||
|
"consent_required": "auto", |
||||
|
"default_consent": True, |
||||
|
"server_action_id": self.update_opt_out.id, |
||||
|
}) |
||||
|
# Activity with manual consent, skipping partner 0 |
||||
|
self.activity_manual = self.env["privacy.activity"].create({ |
||||
|
"name": "activity_manual", |
||||
|
"description": "I'm activity 3", |
||||
|
"subject_find": True, |
||||
|
"subject_domain": repr([("id", "in", self.partners[1:].ids)]), |
||||
|
"consent_required": "manual", |
||||
|
"default_consent": False, |
||||
|
"server_action_id": self.update_opt_out.id, |
||||
|
}) |
||||
|
|
||||
|
# HACK https://github.com/odoo/odoo/issues/12237 |
||||
|
# TODO Remove hack in v12 |
||||
|
def tearDown(self): |
||||
|
self.env = self._oldenv |
||||
|
super(ActivityCase, self).tearDown() |
||||
|
|
||||
|
# HACK https://github.com/odoo/odoo/issues/12237 |
||||
|
# TODO Remove hack in v12 |
||||
|
@contextmanager |
||||
|
def release_cr(self): |
||||
|
self.env.cr.release() |
||||
|
yield |
||||
|
self.env.cr.acquire() |
||||
|
|
||||
|
def check_activity_auto_properly_sent(self): |
||||
|
"""Check emails sent by ``self.activity_auto``.""" |
||||
|
consents = self.env["privacy.consent"].search([ |
||||
|
("activity_id", "=", self.activity_auto.id), |
||||
|
]) |
||||
|
# Check sent mails |
||||
|
for consent in consents: |
||||
|
self.assertEqual(consent.state, "sent") |
||||
|
messages = consent.mapped("message_ids") |
||||
|
self.assertEqual(len(messages), 4) |
||||
|
# 2nd message notifies creation |
||||
|
self.assertEqual( |
||||
|
messages[2].subtype_id, |
||||
|
self.mt_consent_consent_new, |
||||
|
) |
||||
|
# 3rd message notifies subject |
||||
|
# Placeholder links should be logged |
||||
|
self.assertTrue("/privacy/consent/accept/" in messages[1].body) |
||||
|
self.assertTrue("/privacy/consent/reject/" in messages[1].body) |
||||
|
# Tokenized links shouldn't be logged |
||||
|
self.assertFalse(consent._url(True) in messages[1].body) |
||||
|
self.assertFalse(consent._url(False) in messages[1].body) |
||||
|
# 4th message contains the state change |
||||
|
self.assertEqual( |
||||
|
messages[0].subtype_id, |
||||
|
self.mt_consent_state_changed, |
||||
|
) |
||||
|
# Partner's opt_out should be synced with default consent |
||||
|
self.assertFalse(consent.partner_id.opt_out) |
||||
|
|
||||
|
def test_default_template(self): |
||||
|
"""We have a good mail template by default.""" |
||||
|
good = self.env.ref("privacy_consent.template_consent") |
||||
|
self.assertEqual( |
||||
|
self.activity_noconsent.consent_template_id, |
||||
|
good, |
||||
|
) |
||||
|
self.assertEqual( |
||||
|
self.activity_noconsent.consent_template_default_body_html, |
||||
|
good.body_html, |
||||
|
) |
||||
|
self.assertEqual( |
||||
|
self.activity_noconsent.consent_template_default_subject, |
||||
|
good.subject, |
||||
|
) |
||||
|
|
||||
|
def test_find_subject_if_consent_required(self): |
||||
|
"""If user wants to require consent, it needs subjects.""" |
||||
|
# Test the onchange helper |
||||
|
onchange_activity1 = self.env["privacy.activity"].new( |
||||
|
self.activity_noconsent.copy_data()[0]) |
||||
|
self.assertFalse(onchange_activity1.subject_find) |
||||
|
onchange_activity1.consent_required = "auto" |
||||
|
onchange_activity1._onchange_consent_required_subject_find() |
||||
|
self.assertTrue(onchange_activity1.subject_find) |
||||
|
# Test very dumb user that forces an error |
||||
|
with self.assertRaises(ValidationError): |
||||
|
self.activity_noconsent.consent_required = "manual" |
||||
|
|
||||
|
def test_template_required_auto(self): |
||||
|
"""Automatic consent activities need a template.""" |
||||
|
self.activity_noconsent.subject_find = True |
||||
|
self.activity_noconsent.consent_template_id = False |
||||
|
self.activity_noconsent.consent_required = "manual" |
||||
|
with self.assertRaises(ValidationError): |
||||
|
self.activity_noconsent.consent_required = "auto" |
||||
|
|
||||
|
def test_generate_manually(self): |
||||
|
"""Manually-generated consents work as expected.""" |
||||
|
self.partners.write({"opt_out": False}) |
||||
|
result = self.activity_manual.action_new_consents() |
||||
|
self.assertEqual(result["res_model"], "privacy.consent") |
||||
|
consents = self.env[result["res_model"]].search(result["domain"]) |
||||
|
self.assertEqual(consents.mapped("state"), ["draft"] * 2) |
||||
|
self.assertEqual(consents.mapped("partner_id.opt_out"), [False] * 2) |
||||
|
self.assertEqual(consents.mapped("accepted"), [False] * 2) |
||||
|
self.assertEqual(consents.mapped("last_metadata"), [False] * 2) |
||||
|
# Check sent mails |
||||
|
messages = consents.mapped("message_ids") |
||||
|
self.assertEqual(len(messages), 4) |
||||
|
subtypes = messages.mapped("subtype_id") |
||||
|
self.assertTrue(subtypes & self.mt_consent_consent_new) |
||||
|
self.assertFalse(subtypes & self.mt_consent_acceptance_changed) |
||||
|
self.assertFalse(subtypes & self.mt_consent_state_changed) |
||||
|
# Send one manual request |
||||
|
action = consents[0].action_manual_ask() |
||||
|
self.assertEqual(action["res_model"], "mail.compose.message") |
||||
|
composer = self.env[action["res_model"]] \ |
||||
|
.with_context(active_ids=consents[0].ids, |
||||
|
active_model=consents._name, |
||||
|
**action["context"]).create({}) |
||||
|
composer.onchange_template_id_wrapper() |
||||
|
composer.send_mail() |
||||
|
messages = consents.mapped("message_ids") - messages |
||||
|
self.assertEqual(len(messages), 2) |
||||
|
self.assertEqual(messages[0].subtype_id, self.mt_consent_state_changed) |
||||
|
self.assertEqual(consents.mapped("state"), ["sent", "draft"]) |
||||
|
self.assertEqual(consents.mapped("partner_id.opt_out"), [True, False]) |
||||
|
# Placeholder links should be logged |
||||
|
self.assertTrue("/privacy/consent/accept/" in messages[1].body) |
||||
|
self.assertTrue("/privacy/consent/reject/" in messages[1].body) |
||||
|
# Tokenized links shouldn't be logged |
||||
|
accept_url = consents[0]._url(True) |
||||
|
reject_url = consents[0]._url(False) |
||||
|
self.assertNotIn(accept_url, messages[1].body) |
||||
|
self.assertNotIn(reject_url, messages[1].body) |
||||
|
# Visit tokenized accept URL |
||||
|
with self.release_cr(): |
||||
|
result = self.url_open(accept_url).read() |
||||
|
self.assertIn("accepted", result) |
||||
|
self.assertIn(reject_url, result) |
||||
|
self.assertIn(self.activity_manual.name, result) |
||||
|
self.assertIn(self.activity_manual.description, result) |
||||
|
consents.invalidate_cache() |
||||
|
self.assertEqual(consents.mapped("accepted"), [True, False]) |
||||
|
self.assertTrue(consents[0].last_metadata) |
||||
|
self.assertFalse(consents[0].partner_id.opt_out) |
||||
|
self.assertEqual(consents.mapped("state"), ["answered", "draft"]) |
||||
|
self.assertEqual( |
||||
|
consents[0].message_ids[0].subtype_id, |
||||
|
self.mt_consent_acceptance_changed, |
||||
|
) |
||||
|
# Visit tokenized reject URL |
||||
|
with self.release_cr(): |
||||
|
result = self.url_open(reject_url).read() |
||||
|
self.assertIn("rejected", result) |
||||
|
self.assertIn(accept_url, result) |
||||
|
self.assertIn(self.activity_manual.name, result) |
||||
|
self.assertIn(self.activity_manual.description, result) |
||||
|
consents.invalidate_cache() |
||||
|
self.assertEqual(consents.mapped("accepted"), [False, False]) |
||||
|
self.assertTrue(consents[0].last_metadata) |
||||
|
self.assertTrue(consents[0].partner_id.opt_out) |
||||
|
self.assertEqual(consents.mapped("state"), ["answered", "draft"]) |
||||
|
self.assertEqual( |
||||
|
consents[0].message_ids[0].subtype_id, |
||||
|
self.mt_consent_acceptance_changed, |
||||
|
) |
||||
|
self.assertFalse(consents[1].last_metadata) |
||||
|
|
||||
|
def test_generate_automatically(self): |
||||
|
"""Automatically-generated consents work as expected.""" |
||||
|
result = self.activity_auto.action_new_consents() |
||||
|
self.assertEqual(result["res_model"], "privacy.consent") |
||||
|
self.check_activity_auto_properly_sent() |
||||
|
|
||||
|
def test_generate_cron(self): |
||||
|
"""Cron-generated consents work as expected.""" |
||||
|
self.cron.method_direct_trigger() |
||||
|
self.check_activity_auto_properly_sent() |
||||
|
|
||||
|
def test_mail_template_without_links(self): |
||||
|
"""Cannot create mail template without needed links.""" |
||||
|
with self.assertRaises(ValidationError): |
||||
|
self.activity_manual.consent_template_id.body_html = "No links :(" |
@ -0,0 +1,89 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<record id="activity_form" model="ir.ui.view"> |
||||
|
<field name="name">Add consent fields</field> |
||||
|
<field name="model">privacy.activity</field> |
||||
|
<field name="inherit_id" ref="privacy.activity_form"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<div name="button_box" position="inside"> |
||||
|
<!-- TODO Change icon to fa-handshake-o in Odoo 11 --> |
||||
|
<button |
||||
|
attrs='{"invisible": [("consent_required", "=", False)]}' |
||||
|
class="oe_stat_button" |
||||
|
context='{"search_default_activity_id": active_id}' |
||||
|
icon="fa-gavel" |
||||
|
name="%(consent_action)d" |
||||
|
type="action" |
||||
|
> |
||||
|
<field |
||||
|
name="consent_count" |
||||
|
widget="statinfo" |
||||
|
/> |
||||
|
</button> |
||||
|
</div> |
||||
|
|
||||
|
<notebook name="advanced" position="inside"> |
||||
|
<page string="Consent" name="consent"> |
||||
|
<group> |
||||
|
<label for="consent_required"/> |
||||
|
<div> |
||||
|
<field name="consent_required" class="oe_inline"/> |
||||
|
<button |
||||
|
attrs='{"invisible": [("consent_required", "!=", "manual")]}' |
||||
|
class="btn-link" |
||||
|
icon="fa-user-plus" |
||||
|
name="action_new_consents" |
||||
|
type="object" |
||||
|
string="Generate missing draft consent requests" |
||||
|
/> |
||||
|
<button |
||||
|
attrs='{"invisible": [("consent_required", "!=", "auto")]}' |
||||
|
class="btn-link" |
||||
|
icon="fa-user-plus" |
||||
|
name="action_new_consents" |
||||
|
type="object" |
||||
|
string="Generate and send missing consent requests" |
||||
|
confirm="This could send many consent emails, are you sure to proceed?" |
||||
|
/> |
||||
|
</div> |
||||
|
</group> |
||||
|
<group |
||||
|
attrs='{"invisible": [("consent_required", "=", False)]}' |
||||
|
> |
||||
|
<group> |
||||
|
<field name="default_consent"/> |
||||
|
<field |
||||
|
name="server_action_id" |
||||
|
groups="base.group_no_one" |
||||
|
/> |
||||
|
</group> |
||||
|
<group> |
||||
|
<field |
||||
|
name="consent_template_default_body_html" |
||||
|
invisible="1" |
||||
|
/> |
||||
|
<field |
||||
|
name="consent_template_default_subject" |
||||
|
invisible="1" |
||||
|
/> |
||||
|
<field |
||||
|
name="consent_template_id" |
||||
|
attrs='{"required": [("consent_required", "=", "auto")]}' |
||||
|
context='{ |
||||
|
"default_model": "privacy.consent", |
||||
|
"default_body_html": consent_template_default_body_html, |
||||
|
"default_subject": consent_template_default_subject, |
||||
|
}' |
||||
|
/> |
||||
|
</group> |
||||
|
</group> |
||||
|
</page> |
||||
|
</notebook> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
@ -0,0 +1,113 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<record model="ir.ui.view" id="consent_form"> |
||||
|
<field name="name">Privacy Consent Form</field> |
||||
|
<field name="model">privacy.consent</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form> |
||||
|
<header> |
||||
|
<button |
||||
|
type="object" |
||||
|
name="action_manual_ask" |
||||
|
class="oe_highlight" |
||||
|
string="Ask for consent" |
||||
|
/> |
||||
|
<field name="state" widget="statusbar"/> |
||||
|
</header> |
||||
|
<sheet> |
||||
|
<div class="oe_button_box" name="button_box"> |
||||
|
<button |
||||
|
class="oe_stat_button" |
||||
|
icon="fa-archive" |
||||
|
name="toggle_active" |
||||
|
type="object" |
||||
|
> |
||||
|
<field |
||||
|
name="active" |
||||
|
options='{"terminology": "archive"}' |
||||
|
widget="boolean_button" |
||||
|
/> |
||||
|
</button> |
||||
|
</div> |
||||
|
<group> |
||||
|
<field name="partner_id"/> |
||||
|
<field name="activity_id"/> |
||||
|
<field name="accepted"/> |
||||
|
<field name="last_metadata"/> |
||||
|
</group> |
||||
|
</sheet> |
||||
|
<div class="oe_chatter"> |
||||
|
<field name="message_follower_ids" widget="mail_followers"/> |
||||
|
<field name="message_ids" widget="mail_thread"/> |
||||
|
</div> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="consent_tree"> |
||||
|
<field name="name">Privacy Consent Tree</field> |
||||
|
<field name="model">privacy.consent</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<tree> |
||||
|
<field name="activity_id"/> |
||||
|
<field name="partner_id"/> |
||||
|
<field name="state"/> |
||||
|
<field name="accepted"/> |
||||
|
</tree> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record model="ir.ui.view" id="consent_search"> |
||||
|
<field name="name">Privacy Consent Search</field> |
||||
|
<field name="model">privacy.consent</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<search> |
||||
|
<field name="activity_id"/> |
||||
|
<field name="partner_id"/> |
||||
|
<field name="state"/> |
||||
|
<field name="accepted"/> |
||||
|
<separator/> |
||||
|
<filter |
||||
|
string="Archived" |
||||
|
name="inactive" |
||||
|
domain="[('active', '=', False)]" |
||||
|
/> |
||||
|
<separator/> |
||||
|
<group string="Group By" name="groupby"> |
||||
|
<filter |
||||
|
name="activity_id_groupby" |
||||
|
string="Activity" |
||||
|
context="{'group_by': 'activity_id'}" |
||||
|
/> |
||||
|
<filter |
||||
|
name="state_groupby" |
||||
|
string="State" |
||||
|
context="{'group_by': 'state'}" |
||||
|
/> |
||||
|
<filter |
||||
|
name="accepted_groupby" |
||||
|
string="Accepted" |
||||
|
context="{'group_by': 'accepted'}" |
||||
|
/> |
||||
|
</group> |
||||
|
</search> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<act_window |
||||
|
id="consent_action" |
||||
|
name="Consents" |
||||
|
res_model="privacy.consent" |
||||
|
/> |
||||
|
|
||||
|
<menuitem |
||||
|
action="consent_action" |
||||
|
id="menu_privacy_consent" |
||||
|
parent="privacy.menu_data_protection_master_data" |
||||
|
/> |
||||
|
|
||||
|
</data> |
@ -0,0 +1,35 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<!-- Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). --> |
||||
|
|
||||
|
<data> |
||||
|
|
||||
|
<record id="view_partner_form" model="ir.ui.view"> |
||||
|
<field name="name">Add consent smart button</field> |
||||
|
<field name="model">res.partner</field> |
||||
|
<field name="inherit_id" ref="base.view_partner_form"/> |
||||
|
<field |
||||
|
name="groups_id" |
||||
|
eval="[(4, ref('privacy.group_data_protection_user'))]" |
||||
|
/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<div name="button_box" position="inside"> |
||||
|
<!-- TODO Change icon to fa-handshake-o in Odoo 11 --> |
||||
|
<button |
||||
|
attrs='{"invisible": [("privacy_consent_count", "=", 0)]}' |
||||
|
class="oe_stat_button" |
||||
|
context='{"search_default_partner_id": active_id}' |
||||
|
icon="fa-gavel" |
||||
|
name="%(consent_action)d" |
||||
|
type="action" |
||||
|
> |
||||
|
<field |
||||
|
name="privacy_consent_count" |
||||
|
widget="statinfo" |
||||
|
/> |
||||
|
</button> |
||||
|
</div> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</data> |
@ -0,0 +1 @@ |
|||||
|
from . import mail_compose_message |
@ -0,0 +1,16 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2018 Tecnativa - Jairo Llopis |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl). |
||||
|
|
||||
|
from odoo import api, models |
||||
|
|
||||
|
|
||||
|
class MailComposeMessage(models.TransientModel): |
||||
|
_inherit = "mail.compose.message" |
||||
|
|
||||
|
@api.multi |
||||
|
def send_mail(self, auto_commit=False): |
||||
|
"""Force auto commit when sending consent emails.""" |
||||
|
if self.env.context.get('mark_consent_sent'): |
||||
|
auto_commit = True |
||||
|
return super(MailComposeMessage, self).send_mail(auto_commit) |
Write
Preview
Loading…
Cancel
Save
Reference in new issue