2017-07-07 09:13:53 -05:00
|
|
|
# -*- coding: utf-8; -*-
|
2015-12-04 00:31:22 -06:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# Rattail -- Retail Software Framework
|
2018-01-27 19:16:07 -06:00
|
|
|
# Copyright © 2010-2018 Lance Edgar
|
2015-12-04 00:31:22 -06:00
|
|
|
#
|
|
|
|
# This file is part of Rattail.
|
|
|
|
#
|
|
|
|
# Rattail is free software: you can redistribute it and/or modify it under the
|
2017-07-06 23:47:56 -05:00
|
|
|
# terms of the GNU General Public License as published by the Free Software
|
|
|
|
# Foundation, either version 3 of the License, or (at your option) any later
|
|
|
|
# version.
|
2015-12-04 00:31:22 -06:00
|
|
|
#
|
|
|
|
# Rattail is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
2017-07-06 23:47:56 -05:00
|
|
|
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
|
|
# details.
|
2015-12-04 00:31:22 -06:00
|
|
|
#
|
2017-07-06 23:47:56 -05:00
|
|
|
# You should have received a copy of the GNU General Public License along with
|
|
|
|
# Rattail. If not, see <http://www.gnu.org/licenses/>.
|
2015-12-04 00:31:22 -06:00
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
"""
|
|
|
|
Email Views
|
|
|
|
"""
|
|
|
|
|
|
|
|
from __future__ import unicode_literals, absolute_import
|
|
|
|
|
2018-01-27 19:16:07 -06:00
|
|
|
import six
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
from rattail import mail
|
2018-02-06 10:31:44 -06:00
|
|
|
from rattail.db import api, model
|
2016-01-10 17:35:34 -06:00
|
|
|
from rattail.config import parse_list
|
2015-12-04 00:31:22 -06:00
|
|
|
|
2018-01-27 19:16:07 -06:00
|
|
|
import colander
|
|
|
|
from deform import widget as dfwidget
|
2017-07-07 09:13:53 -05:00
|
|
|
from webhelpers2.html import HTML
|
2016-01-10 17:35:34 -06:00
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
from tailbone.db import Session
|
2018-02-05 21:23:23 -06:00
|
|
|
from tailbone.views import View, MasterView
|
2016-01-10 17:35:34 -06:00
|
|
|
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
class ProfilesView(MasterView):
|
|
|
|
"""
|
|
|
|
Master view for email admin (settings/preview).
|
|
|
|
"""
|
|
|
|
normalized_model_name = 'emailprofile'
|
2018-07-18 21:06:07 -05:00
|
|
|
model_title = "Email Setting"
|
2015-12-04 00:31:22 -06:00
|
|
|
model_key = 'key'
|
2018-07-18 21:06:07 -05:00
|
|
|
url_prefix = '/settings/email'
|
2015-12-04 00:31:22 -06:00
|
|
|
filterable = False
|
|
|
|
pageable = False
|
|
|
|
creatable = False
|
|
|
|
deletable = False
|
2018-01-27 19:16:07 -06:00
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
grid_columns = [
|
|
|
|
'key',
|
|
|
|
'prefix',
|
|
|
|
'subject',
|
|
|
|
'to',
|
|
|
|
'enabled',
|
|
|
|
]
|
2015-12-04 00:31:22 -06:00
|
|
|
|
2018-01-27 19:16:07 -06:00
|
|
|
form_fields = [
|
|
|
|
'key',
|
|
|
|
'fallback_key',
|
|
|
|
'description',
|
|
|
|
'prefix',
|
|
|
|
'subject',
|
|
|
|
'sender',
|
|
|
|
'replyto',
|
|
|
|
'to',
|
|
|
|
'cc',
|
|
|
|
'bcc',
|
|
|
|
'enabled',
|
|
|
|
]
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
def get_data(self, session=None):
|
|
|
|
data = []
|
|
|
|
for email in mail.iter_emails(self.rattail_config):
|
|
|
|
key = email.key or email.__name__
|
|
|
|
email = email(self.rattail_config, key)
|
|
|
|
data.append(self.normalize(email))
|
|
|
|
return data
|
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
def configure_grid(self, g):
|
|
|
|
g.sorters['key'] = g.make_simple_sorter('key', foldcase=True)
|
|
|
|
g.sorters['prefix'] = g.make_simple_sorter('prefix', foldcase=True)
|
|
|
|
g.sorters['subject'] = g.make_simple_sorter('subject', foldcase=True)
|
|
|
|
g.sorters['enabled'] = g.make_simple_sorter('enabled')
|
2017-12-04 22:40:10 -06:00
|
|
|
g.set_sort_defaults('key')
|
2017-07-07 09:13:53 -05:00
|
|
|
g.set_type('enabled', 'boolean')
|
|
|
|
g.set_link('key')
|
|
|
|
g.set_link('subject')
|
|
|
|
|
2018-04-01 19:14:00 -05:00
|
|
|
# to
|
|
|
|
g.set_renderer('to', self.render_to_short)
|
|
|
|
g.sorters['to'] = g.make_simple_sorter('to', foldcase=True)
|
|
|
|
|
|
|
|
def render_to_short(self, email, column):
|
|
|
|
profile = email['_email']
|
|
|
|
if self.rattail_config.production():
|
|
|
|
if profile.dynamic_to:
|
|
|
|
if profile.dynamic_to_help:
|
|
|
|
return profile.dynamic_to_help
|
2017-07-07 09:13:53 -05:00
|
|
|
|
2017-08-07 19:09:03 -05:00
|
|
|
value = email['to']
|
|
|
|
if not value:
|
|
|
|
return ""
|
|
|
|
recips = parse_list(value)
|
|
|
|
if len(recips) < 3:
|
|
|
|
return value
|
|
|
|
return "{}, ...".format(', '.join(recips[:2]))
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
def normalize(self, email):
|
2015-12-05 14:39:19 -06:00
|
|
|
def get_recips(type_):
|
|
|
|
recips = email.get_recips(type_)
|
|
|
|
if recips:
|
|
|
|
return ', '.join(recips)
|
2015-12-06 12:17:51 -06:00
|
|
|
data = email.sample_data(self.request)
|
2015-12-04 00:31:22 -06:00
|
|
|
return {
|
2016-08-12 01:17:40 -05:00
|
|
|
'_email': email,
|
2015-12-04 00:31:22 -06:00
|
|
|
'key': email.key,
|
2015-12-05 14:39:19 -06:00
|
|
|
'fallback_key': email.fallback_key,
|
2015-12-04 00:31:22 -06:00
|
|
|
'description': email.__doc__,
|
2018-01-27 19:16:07 -06:00
|
|
|
'prefix': email.get_prefix(data, magic=False) or '',
|
|
|
|
'subject': email.get_subject(data, render=False) or '',
|
|
|
|
'sender': email.get_sender() or '',
|
|
|
|
'replyto': email.get_replyto() or '',
|
|
|
|
'to': get_recips('to') or '',
|
|
|
|
'cc': get_recips('cc') or '',
|
|
|
|
'bcc': get_recips('bcc') or '',
|
2016-08-12 01:17:40 -05:00
|
|
|
'enabled': email.get_enabled(),
|
2015-12-04 00:31:22 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
def get_instance(self):
|
|
|
|
key = self.request.matchdict['key']
|
|
|
|
return self.normalize(mail.get_email(self.rattail_config, key))
|
|
|
|
|
|
|
|
def get_instance_title(self, email):
|
2016-12-01 11:56:26 -06:00
|
|
|
return email['_email'].get_complete_subject(render=False)
|
2015-12-04 00:31:22 -06:00
|
|
|
|
2017-03-19 11:21:00 -05:00
|
|
|
def editable_instance(self, profile):
|
|
|
|
if self.rattail_config.demo():
|
|
|
|
return profile['key'] != 'user_feedback'
|
|
|
|
return True
|
|
|
|
|
|
|
|
def deletable_instance(self, profile):
|
|
|
|
if self.rattail_config.demo():
|
|
|
|
return profile['key'] != 'user_feedback'
|
|
|
|
return True
|
|
|
|
|
2018-01-27 19:16:07 -06:00
|
|
|
def configure_form(self, f):
|
|
|
|
super(ProfilesView, self).configure_form(f)
|
2018-04-01 19:14:00 -05:00
|
|
|
profile = f.model_instance['_email']
|
2018-01-27 19:16:07 -06:00
|
|
|
|
|
|
|
# key
|
|
|
|
f.set_readonly('key')
|
|
|
|
|
|
|
|
# fallback_key
|
|
|
|
f.set_readonly('fallback_key')
|
|
|
|
|
|
|
|
# description
|
|
|
|
f.set_readonly('description')
|
|
|
|
|
|
|
|
# prefix
|
|
|
|
f.set_label('prefix', "Subject Prefix")
|
|
|
|
|
|
|
|
# subject
|
|
|
|
f.set_label('subject', "Subject Text")
|
|
|
|
|
|
|
|
# sender
|
|
|
|
f.set_label('sender', "From")
|
|
|
|
|
|
|
|
# replyto
|
|
|
|
f.set_label('replyto', "Reply-To")
|
|
|
|
|
|
|
|
# to
|
|
|
|
f.set_widget('to', dfwidget.TextAreaWidget(cols=60, rows=6))
|
2018-04-01 19:14:00 -05:00
|
|
|
if self.rattail_config.production():
|
|
|
|
if profile.dynamic_to:
|
|
|
|
f.set_readonly('to')
|
|
|
|
if profile.dynamic_to_help:
|
|
|
|
f.model_instance['to'] = profile.dynamic_to_help
|
2018-01-27 19:16:07 -06:00
|
|
|
|
|
|
|
# cc
|
|
|
|
f.set_widget('cc', dfwidget.TextAreaWidget(cols=60, rows=2))
|
|
|
|
|
|
|
|
# bcc
|
|
|
|
f.set_widget('bcc', dfwidget.TextAreaWidget(cols=60, rows=2))
|
|
|
|
|
|
|
|
# enabled
|
|
|
|
f.set_type('enabled', 'boolean')
|
|
|
|
|
|
|
|
def make_form_schema(self):
|
|
|
|
return EmailProfileSchema()
|
|
|
|
|
|
|
|
def save_edit_form(self, form):
|
|
|
|
key = self.request.matchdict['key']
|
|
|
|
data = self.form_deserialized
|
|
|
|
session = self.Session()
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.prefix'.format(key), data['prefix'])
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.subject'.format(key), data['subject'])
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.from'.format(key), data['sender'])
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.replyto'.format(key), data['replyto'])
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.to'.format(key), (data['to'] or '').replace('\n', ', '))
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.cc'.format(key), (data['cc'] or '').replace('\n', ', '))
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.bcc'.format(key), (data['bcc'] or '').replace('\n', ', '))
|
|
|
|
api.save_setting(session, 'rattail.mail.{}.enabled'.format(key), six.text_type(data['enabled']).lower())
|
|
|
|
return data
|
2016-01-10 17:35:34 -06:00
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
def template_kwargs_view(self, **kwargs):
|
|
|
|
key = self.request.matchdict['key']
|
|
|
|
kwargs['email'] = mail.get_email(self.rattail_config, key)
|
|
|
|
return kwargs
|
|
|
|
|
|
|
|
|
2018-03-12 18:27:50 -05:00
|
|
|
class RecipientsType(colander.String):
|
|
|
|
"""
|
|
|
|
Custom schema type for email recipients. This is used to present the
|
|
|
|
recipients as a "list" within the text area, i.e. one recipient per line.
|
|
|
|
Then the list is collapsed to a comma-delimited string for storage.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def serialize(self, node, appstruct):
|
|
|
|
if appstruct is colander.null:
|
|
|
|
return colander.null
|
|
|
|
recips = parse_list(appstruct)
|
|
|
|
return '\n'.join(recips)
|
|
|
|
|
|
|
|
def deserialize(self, node, cstruct):
|
|
|
|
if cstruct == '' and self.allow_empty:
|
|
|
|
return ''
|
|
|
|
if not cstruct:
|
|
|
|
return colander.null
|
|
|
|
recips = parse_list(cstruct)
|
|
|
|
return ', '.join(recips)
|
|
|
|
|
|
|
|
|
2018-01-27 19:16:07 -06:00
|
|
|
class EmailProfileSchema(colander.MappingSchema):
|
|
|
|
|
|
|
|
prefix = colander.SchemaNode(colander.String())
|
|
|
|
|
|
|
|
subject = colander.SchemaNode(colander.String())
|
|
|
|
|
|
|
|
sender = colander.SchemaNode(colander.String())
|
|
|
|
|
|
|
|
replyto = colander.SchemaNode(colander.String(), missing='')
|
|
|
|
|
2018-03-12 18:27:50 -05:00
|
|
|
to = colander.SchemaNode(RecipientsType())
|
2018-01-27 19:16:07 -06:00
|
|
|
|
2018-03-12 18:27:50 -05:00
|
|
|
cc = colander.SchemaNode(RecipientsType(), missing='')
|
2018-01-27 19:16:07 -06:00
|
|
|
|
2018-03-12 18:27:50 -05:00
|
|
|
bcc = colander.SchemaNode(RecipientsType(), missing='')
|
2018-01-27 19:16:07 -06:00
|
|
|
|
|
|
|
enabled = colander.SchemaNode(colander.Boolean())
|
|
|
|
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
class EmailPreview(View):
|
|
|
|
"""
|
|
|
|
Lists available email templates, and can show previews of each.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __call__(self):
|
|
|
|
|
|
|
|
# Forms submitted via POST are only used for sending emails.
|
|
|
|
if self.request.method == 'POST':
|
|
|
|
self.email_template()
|
2018-01-27 19:16:07 -06:00
|
|
|
url = self.request.get_referrer(default=self.request.route_url('emailprofiles'))
|
|
|
|
return self.redirect(url)
|
2015-12-04 00:31:22 -06:00
|
|
|
|
|
|
|
# Maybe render a preview?
|
|
|
|
key = self.request.GET.get('key')
|
|
|
|
if key:
|
|
|
|
type_ = self.request.GET.get('type', 'html')
|
|
|
|
return self.preview_template(key, type_)
|
|
|
|
|
|
|
|
assert False, "should not be here"
|
|
|
|
|
|
|
|
def email_template(self):
|
|
|
|
recipient = self.request.POST.get('recipient')
|
|
|
|
if recipient:
|
2018-06-28 12:25:44 -05:00
|
|
|
keys = [key for key in self.request.POST.keys()
|
|
|
|
if key.startswith('send_')]
|
2015-12-04 00:31:22 -06:00
|
|
|
key = keys[0][5:] if keys else None
|
|
|
|
if key:
|
|
|
|
email = mail.get_email(self.rattail_config, key)
|
|
|
|
data = email.sample_data(self.request)
|
|
|
|
msg = email.make_message(data)
|
|
|
|
|
|
|
|
subject = msg['Subject']
|
|
|
|
del msg['Subject']
|
|
|
|
msg['Subject'] = "[preview] {0}".format(subject)
|
|
|
|
|
|
|
|
del msg['To']
|
|
|
|
del msg['Cc']
|
|
|
|
del msg['Bcc']
|
|
|
|
msg['To'] = recipient
|
|
|
|
|
2016-12-10 14:54:30 -06:00
|
|
|
sent = mail.deliver_message(self.rattail_config, key, msg)
|
2015-12-04 00:31:22 -06:00
|
|
|
|
2016-12-10 14:54:30 -06:00
|
|
|
self.request.session.flash("Preview for '{}' was {}emailed to {}".format(
|
|
|
|
key, '' if sent else '(NOT) ', recipient))
|
2015-12-04 00:31:22 -06:00
|
|
|
|
|
|
|
def preview_template(self, key, type_):
|
|
|
|
email = mail.get_email(self.rattail_config, key)
|
|
|
|
template = email.get_template(type_)
|
|
|
|
data = email.sample_data(self.request)
|
|
|
|
self.request.response.text = template.render(**data)
|
|
|
|
if type_ == 'txt':
|
|
|
|
self.request.response.content_type = b'text/plain'
|
|
|
|
return self.request.response
|
|
|
|
|
2016-12-01 11:56:26 -06:00
|
|
|
@classmethod
|
|
|
|
def defaults(cls, config):
|
|
|
|
# email preview
|
|
|
|
config.add_route('email.preview', '/email/preview/')
|
|
|
|
config.add_view(cls, route_name='email.preview',
|
|
|
|
renderer='/email/preview.mako',
|
|
|
|
permission='emailprofiles.preview')
|
|
|
|
config.add_tailbone_permission('emailprofiles', 'emailprofiles.preview',
|
|
|
|
"Send preview email")
|
|
|
|
|
|
|
|
|
2018-02-06 10:31:44 -06:00
|
|
|
class EmailAttemptView(MasterView):
|
|
|
|
"""
|
|
|
|
Master view for email attempts.
|
|
|
|
"""
|
|
|
|
model_class = model.EmailAttempt
|
|
|
|
route_prefix = 'email_attempts'
|
|
|
|
url_prefix = '/email/attempts'
|
|
|
|
creatable = False
|
|
|
|
editable = False
|
|
|
|
deletable = False
|
|
|
|
|
|
|
|
labels = {
|
|
|
|
'status_code': "Status",
|
|
|
|
}
|
|
|
|
|
|
|
|
grid_columns = [
|
|
|
|
'key',
|
|
|
|
'sender',
|
|
|
|
'subject',
|
|
|
|
'to',
|
|
|
|
'sent',
|
|
|
|
'status_code',
|
|
|
|
]
|
|
|
|
|
|
|
|
form_fields = [
|
|
|
|
'key',
|
|
|
|
'sender',
|
|
|
|
'subject',
|
|
|
|
'to',
|
|
|
|
'cc',
|
|
|
|
'bcc',
|
|
|
|
'sent',
|
|
|
|
'status_code',
|
|
|
|
'status_text',
|
|
|
|
]
|
|
|
|
|
|
|
|
def configure_grid(self, g):
|
|
|
|
super(EmailAttemptView, self).configure_grid(g)
|
|
|
|
|
|
|
|
# sent
|
|
|
|
g.set_sort_defaults('sent', 'desc')
|
|
|
|
|
|
|
|
# status_code
|
|
|
|
g.set_enum('status_code', self.enum.EMAIL_ATTEMPT)
|
|
|
|
|
|
|
|
# links
|
|
|
|
g.set_link('key')
|
|
|
|
g.set_link('sender')
|
|
|
|
g.set_link('subject')
|
|
|
|
g.set_link('to')
|
|
|
|
|
|
|
|
def configure_form(self, f):
|
|
|
|
super(EmailAttemptView, self).configure_form(f)
|
|
|
|
|
|
|
|
# status_code
|
|
|
|
f.set_enum('status_code', self.enum.EMAIL_ATTEMPT)
|
|
|
|
|
2015-12-04 00:31:22 -06:00
|
|
|
|
|
|
|
def includeme(config):
|
|
|
|
ProfilesView.defaults(config)
|
2016-12-01 11:56:26 -06:00
|
|
|
EmailPreview.defaults(config)
|
2018-02-06 10:31:44 -06:00
|
|
|
EmailAttemptView.defaults(config)
|