Source code for nereid.templating

# The COPYRIGHT file at the top level of this repository contains the full
# copyright notices and license terms.
import email.charset as Charset
import os

from decimal import Decimal
from email import encoders
from email.header import Header
from email.mime.base import MIMEBase
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText

from flask.templating import render_template as flask_render_template
from jinja2 import (  # noqa
    BaseLoader, ChoiceLoader, FileSystemLoader, Template, TemplateNotFound,
    nodes)
from jinja2.ext import Extension
from speaklater import _LazyString

import trytond.tools as tools

from trytond.transaction import Transaction

from .globals import current_app, current_website, request  # noqa
from .helpers import _rst_to_html_filter, make_crumbs

# Override python's weird assumption that utf-8 text should be encoded with
# base64, and instead use quoted-printable (for both subject and body).  I
# can't figure out a way to specify QP (quoted-printable) instead of base64 in
# a way that doesn't modify global state. :-(
#
# wordeology.com/computer/how-to-send-good-unicode-email-with-python.html
Charset.add_charset('utf-8', Charset.QP, Charset.QP, 'utf-8')

# Generally following https://realpython.com/python-send-email/

[docs]class LazyRenderer(_LazyString): """ A Lazy Rendering object which when called renders the template with the current context. >>> lazy_render_object = LazyRenderer('template.html', {'a': 1}) You can change the context by setting values to the contex dictionary >>> lazy_render_object.context['a'] = 100 You can also change the template(s) that should be rendered >>> lazy_render_object.template_name_or_list = "another-template.html" or even, change it into an iterable of templates >>> lazy_render_object.template_name_or_list = ['t1.html', 't2.html'] The template can be rendered and serialized to unicode by calling >>> unicode(lazy_render_object) The status code or header can also be set on the lazy renderer >>> lazy_render_object.sattus = 201 >>> lazy_render_object.headers['X-Some-Header'] = 'header value' .. note:: If the template renders objects which depend on the application, request or a tryton transaction context (like an active record), the call must be made within those contexts. """ __slots__ = ('template_name_or_list', 'context', 'headers', 'status') def __init__( self, template_name_or_list, context, headers=None, eager=False ): """ :param template_name_or_list: the name of the template to be rendered, or an iterable with template names the first one existing will be rendered :param context: the variables that should be available in the context of the template. :param eager: If True a call is made on instantiation to the value to render the template right away with the given context. Useful for debugging. """ self.template_name_or_list = template_name_or_list self.context = context self.headers = {} self.status = 200 if eager: self.render() @property def value(self): """ Return the rendered template with the current context """ return self.render()
[docs] def render(self): """ Return the rendered template with the current context """ return flask_render_template( self.template_name_or_list, **self.context )
def __getstate__(self): return ( self.template_name_or_list, self.context, self.headers, self.status, ) def __setstate__(self, tup): self.template_name_or_list, self.context, \ self.headers, self.status = tup
[docs]def render_template(template_name_or_list, **context): """ Returns a lazy renderer object which renders a template from the template folder with the given context. The returned object is an instance of :class:`LazyRenderer` which has all magic methods implemented to make the object look as close as possible to an unicode object. LazyRenderer objects are automatically converted into response objects by the WSGI dispatcher into a rendered string by the make_response method. :param template_name_or_list: the name of the template to be rendered, or an iterable with template names the first one existing will be rendered :param context: the variables that should be available in the context of the template. """ if current_app.template_prefix_website_name and \ isinstance(template_name_or_list, str): template_name_or_list = [ '/'.join([current_website.name, template_name_or_list]), template_name_or_list ] return LazyRenderer( template_name_or_list, context, eager=current_app.eager_template_render )
def nereid_default_template_ctx_processor(): """Add Decimal and make_crumbs to template context""" return dict( Decimal=Decimal, make_crumbs=make_crumbs, ) NEREID_TEMPLATE_FILTERS = dict( rst=_rst_to_html_filter, )
[docs]class ModuleTemplateLoader(ChoiceLoader): ''' This loader works like the `ChoiceLoader` and loads templates from a filesystem path (optional) followed by the template folders in the tryton module path. The template folders are ordered by the same order in which Tryton arranges modules based on the dependencies. The optional keyword argument searchpath could be used to specify local folder which contains templates which may override the templates bundled into nereid modules. :param database_name: The name of the Tryton database. This is required since the modules installed in a database is what matters and not the modules in the site-packages :param searchpath: Optional filesystem path where templates that override templates bundled with nereid are located. .. versionadded:: 2.8.0.4 .. versionchanged:: 2.8.0.6 Does not accept prefixing of site name anymore ''' def __init__( self, database_name=None, searchpath=None): self.database_name = database_name self.searchpath = searchpath self._loaders = None @property def loaders(self): ''' Lazy load the loaders ''' if self._loaders is None: self._loaders = [] with Transaction().set_user(0) as a, Transaction().reset_context() as b: cursor = Transaction().connection.cursor() cursor.execute( "SELECT name FROM ir_module " "WHERE state = 'activated'" ) activated_modules_list = [name for (name,) in cursor.fetchall()] if self.searchpath is not None: self._loaders.append(FileSystemLoader(self.searchpath)) # Look into the module graph and check if they have template # folders and if they do add them too from trytond.modules import ( EGG_MODULES, MODULES_PATH, create_graph, get_module_list) # Graph in reverse order to provide inheritance with beginning the # search from the last search path to the first. packages = list(create_graph(get_module_list()))[::-1] for package in packages: if package.name not in activated_modules_list: # If the module is not activated in the current database # then don't load the templates either to be consistent # with Tryton's modularity continue if package.name in EGG_MODULES: # trytond.tools has a good helper which allows resources to # be loaded from the installed site packages. Just use it # to load the tryton.cfg file which is guaranteed to exist # and from it lookup the directory. From here, its just # another searchpath for the loader. with tools.file_open(os.path.join(package.name, 'tryton.cfg')) as f: template_dir = os.path.join( os.path.dirname(f.name), 'templates' ) else: template_dir = os.path.join( MODULES_PATH, package.name, 'templates' ) if os.path.isdir(template_dir): # Add to FS Loader only if the folder exists self._loaders.append(FileSystemLoader(template_dir)) return self._loaders
class FragmentCacheExtension(Extension): # a set of names that trigger the extension. tags = set(['cache']) def __init__(self, environment): super(FragmentCacheExtension, self).__init__(environment) # add the defaults to the environment environment.extend( fragment_cache_prefix='', fragment_cache=None ) def parse(self, parser): # the first token is the token that started the tag. In our case # we only listen to ``'cache'`` so this will be a name token with # `cache` as value. We get the line number so that we can give # that line number to the nodes we create by hand. lineno = parser.stream.next().lineno # now we parse a single expression that is used as cache key. args = [parser.parse_expression()] # if there is a comma, the user provided a timeout. If not use # None as second parameter. if parser.stream.skip_if('comma'): args.append(parser.parse_expression()) else: args.append(nodes.Const(None)) # now we parse the body of the cache block up to `endcache` and # drop the needle (which would always be `endcache` in that case) body = parser.parse_statements(['name:endcache'], drop_needle=True) # now return a `CallBlock` node that calls our _cache_support # helper method on this extension. return nodes.CallBlock(self.call_method('_cache_support', args), [], [], body).set_lineno(lineno) def _cache_support(self, name, timeout, caller): """Helper callback.""" key = self.environment.fragment_cache_prefix + name # try to load the block from the cache # if there is no fragment in the cache, render it and store # it in the cache. rv = self.environment.fragment_cache.get(key) if rv is not None: return rv rv = caller() self.environment.fragment_cache.add(key, rv, timeout) return rv
[docs]def render_email( from_email, to, subject, text_template=None, html_template=None, cc=None, attachments=None, **context): """ Read the templates for email messages, format them, construct the email from them and return the corresponding email message object. :param from_email: Email From :param to: Email IDs of direct recepients :param subject: Email subject :param text_template: <Text email template path> :param html_template: <HTML email template path> :param cc: Email IDs of Cc recepients :param attachments: A dict of filename:string as key value pair [preferable file buffer streams] :param context: Context to be sent to template rendering :return: Email multipart instance or Text/HTML part """ if not (text_template or html_template): raise Exception("At least one of HTML or TEXT template is required") text_part = None if text_template: if isinstance(text_template, Template): text = text_template.render(**context) else: text = render_template(text_template, **context) text_part = MIMEText(text.encode("utf-8"), 'plain', _charset="UTF-8") html_part = None if html_template: if isinstance(html_template, Template): html = html_template.render(**context) else: html = render_template(html_template, **context) html_part = MIMEText(html.encode("utf-8"), 'html', _charset="UTF-8") if text_part and html_part: # Construct an alternative part since both the HTML and Text Parts # exist. message = MIMEMultipart('alternative') message.attach(text_part) message.attach(html_part) else: # only one part exists, so use that as the message body. message = text_part or html_part if attachments: # If an attachment exists, the MimeType should be mixed and the # message body should just be another part of it. message_with_attachments = MIMEMultipart('mixed') # Set the message body as the first part message_with_attachments.attach(message) # Now the message _with_attachments itself becomes the message message = message_with_attachments for filename, content in list(attachments.items()): part = MIMEBase('application', "octet-stream") part.set_payload(content) encoders.encode_base64(part) # XXX: Filename might have to be encoded with utf-8, # i.e., part's encoding or with email's encoding part.add_header( 'Content-Disposition', 'attachment; filename="%s"' % filename ) message.attach(part) if isinstance(to, (list, tuple)): to = ', '.join(to) message['Subject'] = Header(str(subject), 'utf-8') message['From'] = from_email message['To'] = to if cc: message['Cc'] = cc return message