admin: add export/import on single mail template (#46094)

This commit is contained in:
Nicolas Roche 2020-12-04 12:46:26 +01:00
parent ab8e860bc3
commit 1d63253d31
4 changed files with 137 additions and 5 deletions

View File

@ -3,8 +3,11 @@
import base64
import os
import pytest
from webtest import Upload
import xml.etree.ElementTree as ET
from django.utils.encoding import force_bytes
from django.utils.six import StringIO
from quixote import cleanup
from wcs.formdef import FormDef
from wcs.fields import FileField
@ -66,6 +69,16 @@ def mail_templates_option(pub):
return pub
@pytest.fixture
def mail_template():
MailTemplate.wipe()
mail_template = MailTemplate(name='test MT')
mail_template.subject = 'test subject'
mail_template.body = 'test body'
mail_template.store()
return mail_template
def test_mail_templates_disabled(pub, superuser):
if not pub.site_options.has_section('options'):
pub.site_options.add_section('options')
@ -301,3 +314,52 @@ def test_workflow_send_mail_template_attachments(pub, superuser, mail_templates_
assert len(payloads) == 3
assert payloads[1].get_content_type() == 'image/jpeg'
assert payloads[2].get_content_type() == 'image/jpeg'
def test_mail_templates_export(pub, superuser, mail_template):
app = login(get_app(pub))
resp = app.get('/backoffice/workflows/mail-templates/1/')
resp = resp.click(href='export')
xml_export = resp.text
ds = StringIO(xml_export)
mail_template2 = MailTemplate.import_from_xml(ds)
assert mail_template2.name == 'test MT'
def test_mail_templates_import(pub, superuser, mail_template):
mail_template.slug = 'foobar'
mail_template.store()
mail_template_xml = ET.tostring(mail_template.export_to_xml(include_id=True))
MailTemplate.wipe()
assert MailTemplate.count() == 0
app = login(get_app(pub))
resp = app.get('/backoffice/workflows/mail-templates/')
resp = resp.click(href='import')
resp.forms[0]['file'] = Upload('mail_template.wcs', mail_template_xml)
resp = resp.forms[0].submit()
assert MailTemplate.count() == 1
assert set([wc.slug for wc in MailTemplate.select()]) == set(['foobar'])
# check slug
resp = app.get('/backoffice/workflows/mail-templates/')
resp = resp.click(href='import')
resp.forms[0]['file'] = Upload('mail_template.wcs', mail_template_xml)
resp = resp.forms[0].submit()
assert MailTemplate.count() == 2
assert set([wc.slug for wc in MailTemplate.select()]) == set(['foobar', 'test-mt'])
resp = app.get('/backoffice/workflows/mail-templates/')
resp = resp.click(href='import')
resp.forms[0]['file'] = Upload('mail_template.wcs', mail_template_xml)
resp = resp.forms[0].submit()
assert MailTemplate.count() == 3
assert set([wc.slug for wc in MailTemplate.select()]) == set(['foobar', 'test-mt', 'test-mt-1'])
# import an invalid file
resp = app.get('/backoffice/workflows/mail-templates/')
resp = resp.click(href='import')
resp.form['file'] = Upload('mail_template.wcs', b'garbage')
resp = resp.form.submit()
assert 'Invalid File' in resp.text

View File

@ -14,19 +14,21 @@
# You should have received a copy of the GNU General Public License
# along with this program; if not, see <http://www.gnu.org/licenses/>.
import xml.etree.ElementTree as ET
from quixote import get_publisher, get_response, redirect
from quixote.directory import Directory
from quixote.html import TemplateIO, htmltext
from wcs.qommon import _, errors, template
from wcs.qommon import _, errors, force_str, misc, template
from wcs.qommon.backoffice.menu import html_top
from wcs.qommon.form import (Form, HtmlWidget, StringWidget, TextWidget,
ComputedExpressionWidget, WidgetList)
ComputedExpressionWidget, WidgetList, FileWidget, get_session)
from wcs.mail_templates import MailTemplate
class MailTemplatesDirectory(Directory):
_q_exports = ['', 'new']
_q_exports = ['', 'new', ('import', 'p_import')]
do_not_call_in_templates = True
def _q_traverse(self, path):
@ -62,9 +64,58 @@ class MailTemplatesDirectory(Directory):
r += form.render()
return r.getvalue()
def p_import(self):
form = Form(enctype='multipart/form-data')
import_title = _('Import Mail Template call')
form.add(FileWidget, 'file', title=_('File'), required=True)
form.add_submit('submit', import_title)
form.add_submit('cancel', _('Cancel'))
if form.get_submit() == 'cancel':
return redirect('.')
if form.is_submitted() and not form.has_errors():
try:
return self.import_submit(form)
except ValueError:
pass
get_response().breadcrumb.append(('import', _('Import')))
html_top('mail_templates', title=import_title)
r = TemplateIO(html=True)
r += htmltext('<h2>%s</h2>') % import_title
r += htmltext('<p>%s</p>') % _(
'You can install a new mail template by uploading a file.')
r += form.render()
return r.getvalue()
def import_submit(self, form):
fp = form.get_widget('file').parse().fp
error = False
try:
mail_template = MailTemplate.import_from_xml(fp)
get_session().message = (
'info', _('This mail template has been successfully imported.'))
except ValueError:
error = True
if error:
form.set_error('file', _('Invalid File'))
raise ValueError()
# check slug unicity
known_slugs = {
x.slug: x.id for x in MailTemplate.select(ignore_migration=True, ignore_errors=True)}
if mail_template.slug in known_slugs:
mail_template.slug = None # a new one will be set in .store()
mail_template.store()
return redirect('%s/' % mail_template.id)
class MailTemplatePage(Directory):
_q_exports = ['', 'edit', 'delete']
_q_exports = ['', 'edit', 'delete', 'export', ]
do_not_call_in_templates = True
def __init__(self, mail_template_id):
@ -74,8 +125,17 @@ class MailTemplatePage(Directory):
raise errors.TraversalError()
get_response().breadcrumb.append((mail_template_id + '/', self.mail_template.name))
def get_sidebar(self):
r = TemplateIO(html=True)
r += htmltext('<ul id="sidebar-actions">')
r += htmltext('<li><a href="export">%s</a></li>') % _('Export')
r += htmltext('<li><a href="delete" rel="popup">%s</a></li>') % _('Delete')
r += htmltext('</ul>')
return r.getvalue()
def _q_index(self):
html_top('mail_templates', title=self.mail_template.name)
get_response().filter['sidebar'] = self.get_sidebar()
return template.QommonTemplateResponse(
templates=['wcs/backoffice/mail-template.html'],
context={'view': self, 'mail_template': self.mail_template})
@ -180,3 +240,13 @@ class MailTemplatePage(Directory):
else:
self.mail_template.remove_self()
return redirect('..')
def export(self):
x = self.mail_template.export_to_xml(include_id=True)
misc.indent_xml(x)
response = get_response()
response.set_content_type('application/x-wcs-mail-template')
response.set_header(
'content-disposition',
'attachment; filename=mail-template-%s.wcs' % self.mail_template.slug)
return '<?xml version="1.0"?>\n' + force_str(ET.tostring(x))

View File

@ -4,7 +4,6 @@
{% block appbar-title %}{% trans "Mail Template" %} - {{ mail_template.name }}{% endblock %}
{% block appbar-actions %}
<a href="delete">{% trans "Delete" %}</a>
<a href="edit">{% trans "Edit" %}</a>
{% endblock %}

View File

@ -4,6 +4,7 @@
{% block appbar-title %}{% trans "Mail Templates" %}{% endblock %}
{% block appbar-actions %}
<a rel="popup" href="import">{% trans "Import" %}</a>
<a rel="popup" href="new">{% trans "New mail template" %}</a>
{% endblock %}