mirror of
https://github.com/bringout/oca-ocb-core.git
synced 2026-04-20 20:32:06 +02:00
19.0 vanilla
This commit is contained in:
parent
d1963a3c3a
commit
2d3ee4855a
7430 changed files with 2687981 additions and 2965473 deletions
|
|
@ -3,11 +3,12 @@
|
|||
|
||||
import ast
|
||||
import re
|
||||
|
||||
from collections import defaultdict
|
||||
from markupsafe import Markup
|
||||
|
||||
from odoo import _, api, fields, models
|
||||
from odoo.exceptions import ValidationError, UserError
|
||||
from odoo.fields import Domain
|
||||
from odoo.tools import is_html_empty, remove_accents
|
||||
|
||||
# see rfc5322 section 3.2.3
|
||||
|
|
@ -15,11 +16,11 @@ atext = r"[a-zA-Z0-9!#$%&'*+\-/=?^_`{|}~]"
|
|||
dot_atom_text = re.compile(r"^%s+(\.%s+)*$" % (atext, atext))
|
||||
|
||||
|
||||
class Alias(models.Model):
|
||||
class MailAlias(models.Model):
|
||||
"""A Mail Alias is a mapping of an email address with a given Odoo Document
|
||||
model. It is used by Odoo's mail gateway when processing incoming emails
|
||||
sent to the system. If the recipient address (To) of the message matches
|
||||
a Mail Alias, the message will be either processed following the rules
|
||||
a Mail MailAlias, the message will be either processed following the rules
|
||||
of that alias. If the message is a reply it will be attached to the
|
||||
existing discussion on the corresponding record, otherwise a new
|
||||
record of the corresponding model will be created.
|
||||
|
|
@ -30,10 +31,20 @@ class Alias(models.Model):
|
|||
"""
|
||||
_name = 'mail.alias'
|
||||
_description = "Email Aliases"
|
||||
_rec_name = 'alias_name'
|
||||
_order = 'alias_model_id, alias_name'
|
||||
_rec_name = 'alias_name'
|
||||
_rec_names_search = ['alias_name', 'alias_domain']
|
||||
|
||||
alias_name = fields.Char('Alias Name', copy=False, help="The name of the email alias, e.g. 'jobs' if you want to catch emails for <jobs@example.odoo.com>")
|
||||
# email definition
|
||||
alias_name = fields.Char(
|
||||
'Alias Name', copy=False,
|
||||
help="The name of the email alias, e.g. 'jobs' if you want to catch emails for <jobs@example.odoo.com>")
|
||||
alias_full_name = fields.Char('Alias Email', compute='_compute_alias_full_name', store=True, index='btree_not_null')
|
||||
alias_domain_id = fields.Many2one(
|
||||
'mail.alias.domain', string='Alias Domain', ondelete='restrict',
|
||||
default=lambda self: self.env.company.alias_domain_id)
|
||||
alias_domain = fields.Char('Alias domain name', related='alias_domain_id.name')
|
||||
# target: create / update
|
||||
alias_model_id = fields.Many2one('ir.model', 'Aliased Model', required=True, ondelete="cascade",
|
||||
help="The model (Odoo Document Kind) to which this alias "
|
||||
"corresponds. Any incoming email that does not reply to an "
|
||||
|
|
@ -42,11 +53,6 @@ class Alias(models.Model):
|
|||
# hack to only allow selecting mail_thread models (we might
|
||||
# (have a few false positives, though)
|
||||
domain="[('field_id.name', '=', 'message_ids')]")
|
||||
alias_user_id = fields.Many2one('res.users', 'Owner', default=lambda self: self.env.user,
|
||||
help="The owner of records created upon receiving emails on this alias. "
|
||||
"If this field is not set the system will attempt to find the right owner "
|
||||
"based on the sender (From) address, or will use the Administrator account "
|
||||
"if no system user is found for that address.")
|
||||
alias_defaults = fields.Text('Default Values', required=True, default='{}',
|
||||
help="A Python dictionary that will be evaluated to provide "
|
||||
"default values when creating new records for this alias.")
|
||||
|
|
@ -54,165 +60,353 @@ class Alias(models.Model):
|
|||
'Record Thread ID',
|
||||
help="Optional ID of a thread (record) to which all incoming messages will be attached, even "
|
||||
"if they did not reply to it. If set, this will disable the creation of new records completely.")
|
||||
alias_domain = fields.Char('Alias domain', compute='_compute_alias_domain')
|
||||
# owner
|
||||
alias_parent_model_id = fields.Many2one(
|
||||
'ir.model', 'Parent Model',
|
||||
help="Parent model holding the alias. The model holding the alias reference "
|
||||
"is not necessarily the model given by alias_model_id "
|
||||
"(example: project (parent_model) and task (model))")
|
||||
alias_parent_thread_id = fields.Integer('Parent Record Thread ID', help="ID of the parent record holding the alias (example: project holding the task creation alias)")
|
||||
alias_contact = fields.Selection([
|
||||
('everyone', 'Everyone'),
|
||||
('partners', 'Authenticated Partners'),
|
||||
('followers', 'Followers only')], default='everyone',
|
||||
alias_parent_thread_id = fields.Integer(
|
||||
'Parent Record Thread ID',
|
||||
help="ID of the parent record holding the alias (example: project holding the task creation alias)")
|
||||
# incoming configuration (mailgateway)
|
||||
alias_contact = fields.Selection(
|
||||
[
|
||||
('everyone', 'Everyone'),
|
||||
('partners', 'Authenticated Partners'),
|
||||
('followers', 'Followers only')
|
||||
], default='everyone',
|
||||
string='Alias Contact Security', required=True,
|
||||
help="Policy to post a message on the document using the mailgateway.\n"
|
||||
"- everyone: everyone can post\n"
|
||||
"- partners: only authenticated partners\n"
|
||||
"- followers: only followers of the related document or members of following channels\n")
|
||||
alias_incoming_local = fields.Boolean('Local-part based incoming detection', default=False)
|
||||
alias_bounced_content = fields.Html(
|
||||
"Custom Bounced Message", translate=True,
|
||||
help="If set, this content will automatically be sent out to unauthorized users instead of the default message.")
|
||||
alias_status = fields.Selection(
|
||||
[
|
||||
('not_tested', 'Not Tested'),
|
||||
('valid', 'Valid'),
|
||||
('invalid', 'Invalid'),
|
||||
], compute='_compute_alias_status', store=True,
|
||||
help='Alias status assessed on the last message received.')
|
||||
|
||||
_sql_constraints = [
|
||||
('alias_unique', 'UNIQUE(alias_name)', 'Unfortunately this email alias is already used, please choose a unique one')
|
||||
]
|
||||
_name_domain_unique = models.UniqueIndex('(alias_name, COALESCE(alias_domain_id, 0))')
|
||||
|
||||
@api.constrains('alias_domain_id', 'alias_force_thread_id', 'alias_parent_model_id',
|
||||
'alias_parent_thread_id', 'alias_model_id')
|
||||
def _check_alias_domain_id_mc(self):
|
||||
""" Check for invalid alias domains based on company configuration.
|
||||
When having a parent record and/or updating an existing record alias
|
||||
domain should match the one used on the related record. """
|
||||
|
||||
# in sudo, to be able to read alias_parent_model_id (ir.model)
|
||||
tocheck = self.sudo().filtered(lambda alias: alias.alias_domain_id.company_ids)
|
||||
# transient check, mainly for tests / install
|
||||
tocheck = tocheck.filtered(lambda alias:
|
||||
(not alias.alias_model_id.model or alias.alias_model_id.model in self.env) and
|
||||
(not alias.alias_parent_model_id.model or alias.alias_parent_model_id.model in self.env)
|
||||
)
|
||||
if not tocheck:
|
||||
return
|
||||
|
||||
# helpers to find owner / target models
|
||||
def _owner_model(alias):
|
||||
return alias.alias_parent_model_id.model
|
||||
def _owner_env(alias):
|
||||
return self.env[_owner_model(alias)]
|
||||
def _target_model(alias):
|
||||
return alias.alias_model_id.model
|
||||
def _target_env(alias):
|
||||
return self.env[_target_model(alias)]
|
||||
|
||||
# fetch impacted records, classify by model
|
||||
recs_by_model = defaultdict(list)
|
||||
for alias in tocheck:
|
||||
# owner record (like 'project.project' for aliases creating new 'project.task')
|
||||
if alias.alias_parent_model_id and alias.alias_parent_thread_id:
|
||||
if _owner_env(alias)._mail_get_company_field():
|
||||
recs_by_model[_owner_model(alias)].append(alias.alias_parent_thread_id)
|
||||
# target record (like 'mail.group' updating a given group)
|
||||
if alias.alias_model_id and alias.alias_force_thread_id:
|
||||
if _target_env(alias)._mail_get_company_field():
|
||||
recs_by_model[_target_model(alias)].append(alias.alias_force_thread_id)
|
||||
|
||||
# helpers to fetch owner / target with prefetching
|
||||
def _fetch_owner(alias):
|
||||
if alias.alias_parent_thread_id in recs_by_model[alias.alias_parent_model_id.model]:
|
||||
return _owner_env(alias).with_prefetch(
|
||||
recs_by_model[_owner_model(alias)]
|
||||
).browse(alias.alias_parent_thread_id)
|
||||
return None
|
||||
def _fetch_target(alias):
|
||||
if alias.alias_force_thread_id in recs_by_model[alias.alias_model_id.model]:
|
||||
return _target_env(alias).with_prefetch(
|
||||
recs_by_model[_target_model(alias)]
|
||||
).browse(alias.alias_force_thread_id)
|
||||
return None
|
||||
|
||||
# check company domains are compatible
|
||||
for alias in tocheck:
|
||||
if owner := _fetch_owner(alias):
|
||||
company = owner[owner._mail_get_company_field()]
|
||||
if company and company.alias_domain_id != alias.alias_domain_id and alias.alias_domain_id.company_ids:
|
||||
raise ValidationError(_(
|
||||
"We could not create alias %(alias_name)s because domain "
|
||||
"%(alias_domain_name)s belongs to company %(alias_company_names)s "
|
||||
"while the owner document belongs to company %(company_name)s.",
|
||||
alias_company_names=','.join(alias.alias_domain_id.company_ids.mapped('name')),
|
||||
alias_domain_name=alias.alias_domain_id.name,
|
||||
alias_name=alias.display_name,
|
||||
company_name=company.name,
|
||||
))
|
||||
if target := _fetch_target(alias):
|
||||
company = target[target._mail_get_company_field()]
|
||||
if company and company.alias_domain_id != alias.alias_domain_id and alias.alias_domain_id.company_ids:
|
||||
raise ValidationError(_(
|
||||
"We could not create alias %(alias_name)s because domain "
|
||||
"%(alias_domain_name)s belongs to company %(alias_company_names)s "
|
||||
"while the target document belongs to company %(company_name)s.",
|
||||
alias_company_names=','.join(alias.alias_domain_id.company_ids.mapped('name')),
|
||||
alias_domain_name=alias.alias_domain_id.name,
|
||||
alias_name=alias.display_name,
|
||||
company_name=company.name,
|
||||
))
|
||||
|
||||
@api.constrains('alias_name')
|
||||
def _alias_is_ascii(self):
|
||||
def _check_alias_is_ascii(self):
|
||||
""" The local-part ("display-name" <local-part@domain>) of an
|
||||
address only contains limited range of ascii characters.
|
||||
We DO NOT allow anything else than ASCII dot-atom formed
|
||||
local-part. Quoted-string and internationnal characters are
|
||||
to be rejected. See rfc5322 sections 3.4.1 and 3.2.3
|
||||
"""
|
||||
for alias in self:
|
||||
if alias.alias_name and not dot_atom_text.match(alias.alias_name):
|
||||
raise ValidationError(_(
|
||||
"You cannot use anything else than unaccented latin characters in the alias address (%s).",
|
||||
alias.alias_name,
|
||||
))
|
||||
|
||||
@api.depends('alias_name')
|
||||
def _compute_alias_domain(self):
|
||||
self.alias_domain = self.env["ir.config_parameter"].sudo().get_param("mail.catchall.domain")
|
||||
for alias in self.filtered('alias_name'):
|
||||
if not dot_atom_text.match(alias.alias_name):
|
||||
raise ValidationError(
|
||||
_("You cannot use anything else than unaccented latin characters in the alias address %(alias_name)s.",
|
||||
alias_name=alias.alias_name)
|
||||
)
|
||||
|
||||
@api.constrains('alias_defaults')
|
||||
def _check_alias_defaults(self):
|
||||
for alias in self:
|
||||
try:
|
||||
dict(ast.literal_eval(alias.alias_defaults))
|
||||
except Exception:
|
||||
raise ValidationError(_('Invalid expression, it must be a literal python dictionary definition e.g. "{\'field\': \'value\'}"'))
|
||||
except Exception as e:
|
||||
raise ValidationError(
|
||||
_('Invalid expression, it must be a literal python dictionary definition e.g. "{\'field\': \'value\'}"')
|
||||
) from e
|
||||
|
||||
@api.constrains('alias_name', 'alias_domain_id')
|
||||
def _check_alias_domain_clash(self):
|
||||
""" Within a given alias domain, aliases should not conflict with bounce
|
||||
or catchall email addresses, as emails should be unique for the gateway. """
|
||||
failing = self.filtered(lambda alias: alias.alias_name and alias.alias_name in [
|
||||
alias.alias_domain_id.bounce_alias, alias.alias_domain_id.catchall_alias
|
||||
])
|
||||
if failing:
|
||||
raise ValidationError(
|
||||
_('Aliases %(alias_names)s is already used as bounce or catchall address. Please choose another alias.',
|
||||
alias_names=', '.join(failing.mapped('display_name')))
|
||||
)
|
||||
|
||||
@api.depends('alias_domain_id.name', 'alias_name')
|
||||
def _compute_alias_full_name(self):
|
||||
""" A bit like display_name, but without the 'inactive alias' UI display.
|
||||
Moreover it is stored, allowing to search on it. """
|
||||
for record in self:
|
||||
if record.alias_domain_id and record.alias_name:
|
||||
record.alias_full_name = f"{record.alias_name}@{record.alias_domain_id.name}"
|
||||
elif record.alias_name:
|
||||
record.alias_full_name = record.alias_name
|
||||
else:
|
||||
record.alias_full_name = False
|
||||
|
||||
@api.depends('alias_domain', 'alias_name')
|
||||
def _compute_display_name(self):
|
||||
""" Return the mail alias display alias_name, including the catchall
|
||||
domain if found otherwise "Inactive Alias". e.g.`jobs@mail.odoo.com`
|
||||
or `jobs` or 'Inactive Alias' """
|
||||
for record in self:
|
||||
if record.alias_name and record.alias_domain:
|
||||
record.display_name = f"{record.alias_name}@{record.alias_domain}"
|
||||
elif record.alias_name:
|
||||
record.display_name = record.alias_name
|
||||
else:
|
||||
record.display_name = _("Inactive Alias")
|
||||
|
||||
@api.depends('alias_contact', 'alias_defaults', 'alias_model_id')
|
||||
def _compute_alias_status(self):
|
||||
"""Reset alias_status to "not_tested" when fields, that can be the source of an error, are modified."""
|
||||
self.alias_status = 'not_tested'
|
||||
|
||||
@api.model_create_multi
|
||||
def create(self, vals_list):
|
||||
""" Creates email.alias records according to the values provided in
|
||||
``vals`` with 1 alteration:
|
||||
""" Creates mail.alias records according to the values provided in
|
||||
``vals`` but sanitize 'alias_name' by replacing certain unsafe
|
||||
characters; set default alias domain if not given.
|
||||
|
||||
* ``alias_name`` value may be cleaned by replacing certain unsafe
|
||||
characters;
|
||||
|
||||
:raise UserError: if given alias_name is already assigned or there are
|
||||
duplicates in given vals_list;
|
||||
:raise UserError: if given (alias_name, alias_domain_id) already exists
|
||||
or if there are duplicates in given vals_list;
|
||||
"""
|
||||
alias_names = [vals['alias_name'] for vals in vals_list if vals.get('alias_name')]
|
||||
if alias_names:
|
||||
sanitized_names = self._clean_and_check_unique(alias_names)
|
||||
for vals in vals_list:
|
||||
if vals.get('alias_name'):
|
||||
vals['alias_name'] = sanitized_names[alias_names.index(vals['alias_name'])]
|
||||
return super(Alias, self).create(vals_list)
|
||||
alias_names, alias_domains = [], []
|
||||
for vals in vals_list:
|
||||
vals['alias_name'] = self._sanitize_alias_name(vals.get('alias_name'))
|
||||
alias_names.append(vals['alias_name'])
|
||||
vals['alias_domain_id'] = vals.get('alias_domain_id', self.env.company.alias_domain_id.id)
|
||||
alias_domains.append(self.env['mail.alias.domain'].browse(vals['alias_domain_id']))
|
||||
|
||||
self._check_unique(alias_names, alias_domains)
|
||||
return super().create(vals_list)
|
||||
|
||||
def write(self, vals):
|
||||
""""Raises UserError if given alias name is already assigned"""
|
||||
if vals.get('alias_name') and self.ids:
|
||||
if len(self) > 1:
|
||||
raise UserError(_(
|
||||
'Email alias %(alias_name)s cannot be used on %(count)d records at the same time. Please update records one by one.',
|
||||
alias_name=vals['alias_name'], count=len(self)
|
||||
))
|
||||
vals['alias_name'] = self._clean_and_check_unique([vals.get('alias_name')])[0]
|
||||
return super(Alias, self).write(vals)
|
||||
|
||||
def name_get(self):
|
||||
"""Return the mail alias display alias_name, including the implicit
|
||||
mail catchall domain if exists from config otherwise "New Alias".
|
||||
e.g. `jobs@mail.odoo.com` or `jobs` or 'New Alias'
|
||||
""" Raise UserError with a meaningful message instead of letting the
|
||||
uniqueness constraint raise an SQL error. To check uniqueness we have
|
||||
to rebuild pairs of names / domains to validate, taking into account
|
||||
that a void alias_domain_id is acceptable (but also raises for
|
||||
uniqueness).
|
||||
"""
|
||||
res = []
|
||||
for record in self:
|
||||
if record.alias_name and record.alias_domain:
|
||||
res.append((record['id'], "%s@%s" % (record.alias_name, record.alias_domain)))
|
||||
elif record.alias_name:
|
||||
res.append((record['id'], "%s" % (record.alias_name)))
|
||||
alias_names, alias_domains = [], []
|
||||
if 'alias_name' in vals:
|
||||
vals['alias_name'] = self._sanitize_alias_name(vals['alias_name'])
|
||||
if vals.get('alias_name') and self.ids:
|
||||
alias_names = [vals['alias_name']] * len(self)
|
||||
elif 'alias_name' not in vals and 'alias_domain_id' in vals:
|
||||
# avoid checking when writing the same value
|
||||
if [vals['alias_domain_id']] != self.alias_domain_id.ids:
|
||||
alias_names = self.filtered('alias_name').mapped('alias_name')
|
||||
|
||||
if alias_names:
|
||||
tocheck_records = self if vals.get('alias_name') else self.filtered('alias_name')
|
||||
if 'alias_domain_id' in vals:
|
||||
alias_domains = [self.env['mail.alias.domain'].browse(vals['alias_domain_id'])] * len(tocheck_records)
|
||||
else:
|
||||
res.append((record['id'], _("Inactive Alias")))
|
||||
return res
|
||||
alias_domains = [record.alias_domain_id for record in tocheck_records]
|
||||
self._check_unique(alias_names, alias_domains)
|
||||
|
||||
def _clean_and_check_mail_catchall_allowed_domains(self, value):
|
||||
""" The purpose of this system parameter is to avoid the creation
|
||||
of records from incoming emails with a domain != alias_domain
|
||||
but that have a pattern matching an internal mail.alias . """
|
||||
value = [domain.strip().lower() for domain in value.split(',') if domain.strip()]
|
||||
if not value:
|
||||
raise ValidationError(_("Value for `mail.catchall.domain.allowed` cannot be validated.\n"
|
||||
"It should be a comma separated list of domains e.g. example.com,example.org."))
|
||||
return ",".join(value)
|
||||
return super().write(vals)
|
||||
|
||||
def _clean_and_check_unique(self, names):
|
||||
"""When an alias name appears to already be an email, we keep the local
|
||||
part only. A sanitizing / cleaning is also performed on the name. If
|
||||
name already exists an UserError is raised. """
|
||||
def _check_unique(self, alias_names, alias_domains):
|
||||
""" Check unicity constraint won't be raised, otherwise raise a UserError
|
||||
with a complete error message. Also check unicity against alias config
|
||||
parameters.
|
||||
|
||||
def _sanitize_alias_name(name):
|
||||
""" Cleans and sanitizes the alias name """
|
||||
sanitized_name = remove_accents(name).lower().split('@')[0]
|
||||
sanitized_name = re.sub(r'[^\w+.]+', '-', sanitized_name)
|
||||
sanitized_name = re.sub(r'^\.+|\.+$|\.+(?=\.)', '', sanitized_name)
|
||||
sanitized_name = sanitized_name.encode('ascii', errors='replace').decode()
|
||||
return sanitized_name
|
||||
:param list alias_names: a list of names (considered as sanitized
|
||||
and ready to be sent to DB);
|
||||
:param list alias_domains: list of alias_domain records under which
|
||||
the check is performed, as uniqueness is performed for given pair
|
||||
(name, alias_domain);
|
||||
"""
|
||||
if len(alias_names) != len(alias_domains):
|
||||
msg = (f"Invalid call to '_check_unique': names and domains should make coherent lists, "
|
||||
f"received {', '.join(alias_names)} and {', '.join(alias_domains.mapped('name'))}")
|
||||
raise ValueError(msg)
|
||||
|
||||
sanitized_names = [_sanitize_alias_name(name) for name in names]
|
||||
|
||||
catchall_alias = self.env['ir.config_parameter'].sudo().get_param('mail.catchall.alias')
|
||||
bounce_alias = self.env['ir.config_parameter'].sudo().get_param('mail.bounce.alias')
|
||||
alias_domain = self.env["ir.config_parameter"].sudo().get_param("mail.catchall.domain")
|
||||
|
||||
# matches catchall or bounce alias
|
||||
for sanitized_name in sanitized_names:
|
||||
if sanitized_name in [catchall_alias, bounce_alias]:
|
||||
matching_alias_name = '%s@%s' % (sanitized_name, alias_domain) if alias_domain else sanitized_name
|
||||
# reorder per alias domain, keep only not void alias names (void domain also checks uniqueness)
|
||||
domain_to_names = defaultdict(list)
|
||||
for alias_name, alias_domain in zip(alias_names, alias_domains):
|
||||
if alias_name and alias_name in domain_to_names[alias_domain]:
|
||||
raise UserError(
|
||||
_('The e-mail alias %(matching_alias_name)s is already used as %(alias_duplicate)s alias. Please choose another alias.',
|
||||
matching_alias_name=matching_alias_name,
|
||||
alias_duplicate=_('catchall') if sanitized_name == catchall_alias else _('bounce'))
|
||||
_('Email aliases %(alias_name)s cannot be used on several records at the same time. Please update records one by one.',
|
||||
alias_name=alias_name)
|
||||
)
|
||||
if alias_name:
|
||||
domain_to_names[alias_domain].append(alias_name)
|
||||
|
||||
# matches existing alias
|
||||
domain = [('alias_name', 'in', sanitized_names)]
|
||||
if self:
|
||||
domain += [('id', 'not in', self.ids)]
|
||||
matching_alias = self.search(domain, limit=1)
|
||||
if not matching_alias:
|
||||
return sanitized_names
|
||||
|
||||
sanitized_alias_name = _sanitize_alias_name(matching_alias.alias_name)
|
||||
matching_alias_name = '%s@%s' % (sanitized_alias_name, alias_domain) if alias_domain else sanitized_alias_name
|
||||
if matching_alias.alias_parent_model_id and matching_alias.alias_parent_thread_id:
|
||||
# If parent model and parent thread ID both are set, display document name also in the warning
|
||||
document_name = self.env[matching_alias.alias_parent_model_id.model].sudo().browse(matching_alias.alias_parent_thread_id).display_name
|
||||
raise UserError(
|
||||
_('The e-mail alias %(matching_alias_name)s is already used by the %(document_name)s %(model_name)s. Choose another alias or change it on the other document.',
|
||||
matching_alias_name=matching_alias_name,
|
||||
document_name=document_name,
|
||||
model_name=matching_alias.alias_parent_model_id.name)
|
||||
)
|
||||
raise UserError(
|
||||
_('The e-mail alias %(matching_alias_name)s is already linked with %(alias_model_name)s. Choose another alias or change it on the linked model.',
|
||||
matching_alias_name=matching_alias_name,
|
||||
alias_model_name=matching_alias.alias_model_id.name)
|
||||
domain = Domain.OR(
|
||||
Domain('alias_name', 'in', alias_names) & Domain('alias_domain_id', '=', alias_domain.id)
|
||||
for alias_domain, alias_names in domain_to_names.items()
|
||||
)
|
||||
if domain and self:
|
||||
domain &= Domain('id', 'not in', self.ids)
|
||||
existing = self.search(domain, limit=1) if domain else self.env['mail.alias']
|
||||
if not existing:
|
||||
return
|
||||
if existing.alias_parent_model_id and existing.alias_parent_thread_id:
|
||||
parent_name = self.env[existing.alias_parent_model_id.model].sudo().browse(existing.alias_parent_thread_id).display_name
|
||||
msg_begin = _(
|
||||
'Alias %(matching_name)s (%(current_id)s) is already linked with %(alias_model_name)s (%(matching_id)s) and used by the %(parent_name)s %(parent_model_name)s.',
|
||||
alias_model_name=existing.alias_model_id.name,
|
||||
current_id=self.ids if self else _('your alias'),
|
||||
matching_id=existing.id,
|
||||
matching_name=existing.display_name,
|
||||
parent_name=parent_name,
|
||||
parent_model_name=existing.alias_parent_model_id.name
|
||||
)
|
||||
else:
|
||||
msg_begin = _(
|
||||
'Alias %(matching_name)s (%(current_id)s) is already linked with %(alias_model_name)s (%(matching_id)s).',
|
||||
alias_model_name=existing.alias_model_id.name,
|
||||
current_id=self.ids if self else _('new'),
|
||||
matching_id=existing.id,
|
||||
matching_name=existing.display_name,
|
||||
)
|
||||
msg_end = _('Choose another value or change it on the other document.')
|
||||
raise UserError(f'{msg_begin} {msg_end}') # pylint: disable=missing-gettext
|
||||
|
||||
@api.model
|
||||
def _sanitize_allowed_domains(self, allowed_domains):
|
||||
""" When having aliases checked on email left-part only we may define
|
||||
an allowed list for right-part filtering, allowing more fine-grain than
|
||||
either alias domain, either everything. This method sanitized its value. """
|
||||
value = [domain.strip().lower() for domain in allowed_domains.split(',') if domain.strip()]
|
||||
if not value:
|
||||
raise ValidationError(_(
|
||||
"Value %(allowed_domains)s for `mail.catchall.domain.allowed` cannot be validated.\n"
|
||||
"It should be a comma separated list of domains e.g. example.com,example.org.",
|
||||
allowed_domains=allowed_domains
|
||||
))
|
||||
return ",".join(value)
|
||||
|
||||
@api.model
|
||||
def _sanitize_alias_name(self, name, is_email=False):
|
||||
""" Cleans and sanitizes the alias name. In some cases we want the alias
|
||||
to be a complete email instead of just a left-part (when sanitizing
|
||||
default.from for example). In that case we extract the right part and
|
||||
put it back after sanitizing the left part.
|
||||
|
||||
:param str name: the alias name to sanitize;
|
||||
:param bool is_email: whether to keep a right part, otherwise only
|
||||
left part is kept;
|
||||
|
||||
:returns: sanitized alias name
|
||||
:rtype: str
|
||||
"""
|
||||
sanitized_name = name.strip() if name else ''
|
||||
if is_email:
|
||||
right_part = sanitized_name.lower().partition('@')[2]
|
||||
else:
|
||||
right_part = False
|
||||
if sanitized_name:
|
||||
sanitized_name = remove_accents(sanitized_name).lower().split('@')[0]
|
||||
# cannot start and end with dot
|
||||
sanitized_name = re.sub(r'^\.+|\.+$|\.+(?=\.)', '', sanitized_name)
|
||||
# subset of allowed characters
|
||||
sanitized_name = re.sub(r'[^\w!#$%&\'*+\-/=?^_`{|}~.]+', '-', sanitized_name)
|
||||
sanitized_name = sanitized_name.encode('ascii', errors='replace').decode()
|
||||
if not sanitized_name.strip():
|
||||
return False
|
||||
return f'{sanitized_name}@{right_part}' if is_email and right_part else sanitized_name
|
||||
|
||||
@api.model
|
||||
def _is_encodable(self, alias_name, charset='ascii'):
|
||||
""" Check if alias_name is encodable. Standard charset is ascii, as
|
||||
UTF-8 requires a specific extension. Not recommended for outgoing
|
||||
aliases. 'remove_accents' is performed as sanitization process of
|
||||
the name will do it anyway. """
|
||||
try:
|
||||
remove_accents(alias_name).encode(charset)
|
||||
except UnicodeEncodeError:
|
||||
return False
|
||||
return True
|
||||
|
||||
# ------------------------------------------------------------
|
||||
# ACTIONS
|
||||
# ------------------------------------------------------------
|
||||
|
||||
def open_document(self):
|
||||
if not self.alias_model_id or not self.alias_force_thread_id:
|
||||
|
|
@ -234,36 +428,23 @@ class Alias(models.Model):
|
|||
'type': 'ir.actions.act_window',
|
||||
}
|
||||
|
||||
def _get_alias_bounced_body_fallback(self, message_dict):
|
||||
contact_description = self._get_alias_contact_description()
|
||||
default_email = self.env.company.partner_id.email_formatted if self.env.company.partner_id.email else self.env.company.name
|
||||
return Markup(
|
||||
_("""<p>Dear Sender,<br /><br />
|
||||
The message below could not be accepted by the address %(alias_display_name)s.
|
||||
Only %(contact_description)s are allowed to contact it.<br /><br />
|
||||
Please make sure you are using the correct address or contact us at %(default_email)s instead.<br /><br />
|
||||
Kind Regards,</p>"""
|
||||
)) % {
|
||||
'alias_display_name': self.display_name,
|
||||
'contact_description': contact_description,
|
||||
'default_email': default_email,
|
||||
}
|
||||
|
||||
def _get_alias_contact_description(self):
|
||||
if self.alias_contact == 'partners':
|
||||
return _('addresses linked to registered partners')
|
||||
return _('some specific addresses')
|
||||
# ------------------------------------------------------------
|
||||
# MAIL GATEWAY
|
||||
# ------------------------------------------------------------
|
||||
|
||||
def _get_alias_bounced_body(self, message_dict):
|
||||
"""Get the body of the email return in case of bounced email.
|
||||
"""Get the body of the email return in case of bounced email when the
|
||||
alias does not accept incoming email e.g. contact is not allowed.
|
||||
|
||||
:param message_dict: dictionary of mail values
|
||||
:param dict message_dict: dictionary holding parsed message variables
|
||||
|
||||
:return: HTML to use as email body
|
||||
"""
|
||||
lang_author = False
|
||||
if message_dict.get('author_id'):
|
||||
try:
|
||||
lang_author = self.env['res.partner'].browse(message_dict['author_id']).lang
|
||||
except:
|
||||
except Exception:
|
||||
pass
|
||||
|
||||
if lang_author:
|
||||
|
|
@ -277,3 +458,77 @@ Kind Regards,</p>"""
|
|||
'body': body,
|
||||
'message': message_dict
|
||||
}, minimal_qcontext=True)
|
||||
|
||||
def _get_alias_bounced_body_fallback(self, message_dict):
|
||||
""" Default body of bounced emails. See '_get_alias_bounced_body' """
|
||||
contact_description = self._get_alias_contact_description()
|
||||
default_email = self.env.company.partner_id.email_formatted if self.env.company.partner_id.email else self.env.company.name
|
||||
content = Markup(
|
||||
_("""The message below could not be accepted by the address %(alias_display_name)s.
|
||||
Only %(contact_description)s are allowed to contact it.<br /><br />
|
||||
Please make sure you are using the correct address or contact us at %(default_email)s instead."""
|
||||
)
|
||||
) % {
|
||||
'alias_display_name': self.display_name,
|
||||
'contact_description': contact_description,
|
||||
'default_email': default_email,
|
||||
}
|
||||
return Markup('<p>%(header)s,<br /><br />%(content)s<br /><br />%(regards)s</p>') % {
|
||||
'content': content,
|
||||
'header': _('Dear Sender'),
|
||||
'regards': _('Kind Regards'),
|
||||
}
|
||||
|
||||
def _get_alias_contact_description(self):
|
||||
if self.alias_contact == 'partners':
|
||||
return _('addresses linked to registered partners')
|
||||
return _('some specific addresses')
|
||||
|
||||
def _get_alias_invalid_body(self, message_dict):
|
||||
"""Get the body of the bounced email returned when the alias is incorrectly
|
||||
configured e.g. error in alias_defaults.
|
||||
|
||||
:param dict message_dict: dictionary holding parsed message variables
|
||||
|
||||
:return: HTML to use as email body
|
||||
"""
|
||||
content = Markup(
|
||||
_("""The message below could not be accepted by the address %(alias_display_name)s.
|
||||
Please try again later or contact %(company_name)s instead."""
|
||||
)
|
||||
) % {
|
||||
'alias_display_name': self.display_name,
|
||||
'company_name': self.env.company.name,
|
||||
}
|
||||
return self.env['ir.qweb']._render('mail.mail_bounce_alias_security', {
|
||||
'body': Markup('<p>%(header)s,<br /><br />%(content)s<br /><br />%(regards)s</p>') % {
|
||||
'content': content,
|
||||
'header': _('Dear Sender'),
|
||||
'regards': _('Kind Regards'),
|
||||
},
|
||||
'message': message_dict
|
||||
}, minimal_qcontext=True)
|
||||
|
||||
def _alias_bounce_incoming_email(self, message, message_dict, set_invalid=True):
|
||||
"""Set alias status to invalid and create bounce message to the sender.
|
||||
|
||||
This method must be called when a message received on the alias has
|
||||
caused an error due to the mis-configuration of the alias.
|
||||
|
||||
:param EmailMessage message: email message that is invalid and is about
|
||||
to bounce;
|
||||
:param dict message_dict: dictionary holding parsed message variables
|
||||
:param bool set_invalid: set alias as invalid, to be done notably if
|
||||
bounce is considered as coming from a configuration error instead of
|
||||
being rejected due to alias rules;
|
||||
"""
|
||||
self.ensure_one()
|
||||
if set_invalid:
|
||||
self.alias_status = 'invalid'
|
||||
body = self._get_alias_invalid_body(message_dict)
|
||||
else:
|
||||
body = self._get_alias_bounced_body(message_dict)
|
||||
self.env['mail.thread']._routing_create_bounce_email(
|
||||
message_dict['email_from'], body, message,
|
||||
references=message_dict['message_id'],
|
||||
)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue