mirror of
https://github.com/explosion/spaCy.git
synced 2024-11-14 13:47:13 +03:00
e2f2ef3a5a
- As much as I dislike YAML, it seemed like a better format here because it allows us to add comments if we want to explain the different recommendations - Don't include the generated JS in the repo by default and build it on the fly when running or deploying the site. This ensures it's always up to date. - Simplify jinja_to_js script and use fewer dependencies
1274 lines
46 KiB
Python
1274 lines
46 KiB
Python
# Forked from: https://github.com/jonbretman/jinja-to-js
|
|
# With additional functionality: in/not in, replace, pprint, round, + for lists,
|
|
# rendering empty dicts
|
|
# This script is mostly used to generate the JavaScript function for the
|
|
# training quickstart widget.
|
|
import contextlib
|
|
import json
|
|
import re
|
|
import os
|
|
from os import path
|
|
from io import StringIO
|
|
from jinja2 import Environment, FileSystemLoader, nodes
|
|
from pathlib import Path
|
|
import srsly
|
|
import sys
|
|
|
|
|
|
OPERANDS = {
|
|
"eq": "===",
|
|
"ne": "!==",
|
|
"lt": " < ",
|
|
"gt": " > ",
|
|
"lteq": " <= ",
|
|
"gteq": " >= ",
|
|
}
|
|
|
|
DICT_ITER_METHODS = ("iteritems", "items", "values", "keys")
|
|
|
|
STATE_DEFAULT = 0
|
|
STATE_EXECUTING = 1
|
|
STATE_INTERPOLATING = 2
|
|
|
|
LOOP_HELPER_INDEX = "index"
|
|
LOOP_HELPER_INDEX_0 = "index0"
|
|
LOOP_HELPER_FIRST = "first"
|
|
LOOP_HELPER_LAST = "last"
|
|
LOOP_HELPER_LENGTH = "length"
|
|
LOOP_HELPERS = (
|
|
LOOP_HELPER_INDEX,
|
|
LOOP_HELPER_INDEX_0,
|
|
LOOP_HELPER_FIRST,
|
|
LOOP_HELPER_LAST,
|
|
LOOP_HELPER_LENGTH,
|
|
)
|
|
|
|
|
|
def amd_format(dependencies, template_function):
|
|
result = "define(["
|
|
result += ",".join('"{0}"'.format(x[0]) for x in dependencies)
|
|
result += "], function ("
|
|
result += ",".join(x[1] for x in dependencies)
|
|
result += ") { return "
|
|
result += template_function
|
|
result += "; });"
|
|
return result
|
|
|
|
|
|
def commonjs_format(dependencies, template_function):
|
|
result = "".join('var {0} = require("{1}");'.format(y, x) for x, y in dependencies)
|
|
result += "module.exports = {0};".format(template_function)
|
|
return result
|
|
|
|
|
|
def es6_format(dependencies, template_function):
|
|
result = "".join('import {0} from "{1}";'.format(y, x) for x, y in dependencies)
|
|
result += "export default {0}".format(template_function)
|
|
return result
|
|
|
|
|
|
JS_MODULE_FORMATS = {
|
|
None: lambda dependencies, template_function: template_function,
|
|
"amd": amd_format,
|
|
"commonjs": commonjs_format,
|
|
"es6": es6_format,
|
|
}
|
|
|
|
|
|
# This string has to double all the '{' and '}' due to Python's string formatting.
|
|
# See - https://docs.python.org/2/library/string.html#formatstrings
|
|
TEMPLATE_WRAPPER = """
|
|
function {function_name}(ctx) {{
|
|
var __result = "";
|
|
var __tmp;
|
|
var __runtime = jinjaToJS.runtime;
|
|
var __filters = jinjaToJS.filters;
|
|
var __globals = jinjaToJS.globals;
|
|
var context = jinjaToJS.createContext(ctx);
|
|
{template_code}
|
|
return __result;
|
|
}}
|
|
"""
|
|
|
|
|
|
class ExtendsException(Exception):
|
|
"""
|
|
Raised when an {% extends %} is encountered. At this point the parent template is
|
|
loaded and all blocks defined in the current template passed to it.
|
|
"""
|
|
|
|
pass
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def option(current_kwargs, **kwargs):
|
|
"""
|
|
Context manager for temporarily setting a keyword argument and
|
|
then restoring it to whatever it was before.
|
|
"""
|
|
|
|
tmp_kwargs = dict((key, current_kwargs.get(key)) for key, value in kwargs.items())
|
|
current_kwargs.update(kwargs)
|
|
yield
|
|
current_kwargs.update(tmp_kwargs)
|
|
|
|
|
|
def is_method_call(node, method_name):
|
|
"""
|
|
Returns True if `node` is a method call for `method_name`. `method_name`
|
|
can be either a string or an iterable of strings.
|
|
"""
|
|
|
|
if not isinstance(node, nodes.Call):
|
|
return False
|
|
|
|
if isinstance(node.node, nodes.Getattr):
|
|
# e.g. foo.bar()
|
|
method = node.node.attr
|
|
|
|
elif isinstance(node.node, nodes.Name):
|
|
# e.g. bar()
|
|
method = node.node.name
|
|
|
|
elif isinstance(node.node, nodes.Getitem):
|
|
# e.g. foo["bar"]()
|
|
method = node.node.arg.value
|
|
|
|
else:
|
|
return False
|
|
|
|
if isinstance(method_name, (list, tuple)):
|
|
return method in method_name
|
|
|
|
return method == method_name
|
|
|
|
|
|
def is_loop_helper(node):
|
|
"""
|
|
Returns True is node is a loop helper e.g. {{ loop.index }} or {{ loop.first }}
|
|
"""
|
|
return (
|
|
hasattr(node, "node")
|
|
and isinstance(node.node, nodes.Name)
|
|
and node.node.name == "loop"
|
|
)
|
|
|
|
|
|
def temp_var_names_generator():
|
|
x = 0
|
|
while True:
|
|
yield "__$%s" % x
|
|
x += 1
|
|
|
|
|
|
class JinjaToJS(object):
|
|
def __init__(
|
|
self,
|
|
template_root,
|
|
template_name,
|
|
js_module_format=None,
|
|
runtime_path="jinja-to-js",
|
|
include_prefix="",
|
|
include_ext="",
|
|
child_blocks=None,
|
|
dependencies=None,
|
|
custom_filters=None,
|
|
):
|
|
"""
|
|
Args:
|
|
template_root (str): The path to where templates should be loaded from.
|
|
template_name (str): The name of the template to compile (relative to `template_root`).
|
|
js_module_format (str, optional): The JavaScript module format to use.
|
|
One of ('amd', 'commonjs', 'es6')
|
|
runtime_path (str, optional): If `js_module_format` is specified then the JavaScript
|
|
runtime will be imported using the appropriate method.
|
|
It defaults to assuming it will be imported from
|
|
`node_modules` but you can change it using this option.
|
|
include_prefix (str, optional): If using the `amd` module format you can use this option
|
|
to add a prefix to every include path as AMD imports are
|
|
generally relative to the main file, not the module
|
|
importing.
|
|
include_ext (str, optional): By default any includes will be references without an
|
|
extension, as neither AMD, commonJS or ES6 require the
|
|
'.js' extension. If you want to use an extension, say
|
|
'.template' then set this option to a string including
|
|
the leading '.'
|
|
child_blocks (dict, optional): Used internally when handling templates that extend
|
|
other templates.
|
|
dependencies (list of tuple, optional): Used internally when handling templates that
|
|
extend other templates.
|
|
custom_filters (list of str, optional): List of custom filters which should be allowed.
|
|
These may be filters supported by Jinja but not
|
|
supported by jinja-to-js. These filters MUST be
|
|
registered with the jinja-to-js JS runtime.
|
|
"""
|
|
|
|
self.environment = Environment(
|
|
loader=FileSystemLoader(template_root),
|
|
autoescape=True,
|
|
extensions=["jinja2.ext.with_", "jinja2.ext.autoescape"],
|
|
)
|
|
self.output = StringIO()
|
|
self.stored_names = set()
|
|
self.temp_var_names = temp_var_names_generator()
|
|
self.state = STATE_DEFAULT
|
|
self.child_blocks = child_blocks or {}
|
|
self.dependencies = dependencies or []
|
|
self._runtime_function_cache = []
|
|
self.js_module_format = js_module_format
|
|
self.runtime_path = runtime_path
|
|
self.include_prefix = include_prefix
|
|
self.include_ext = include_ext
|
|
self.template_root = template_root
|
|
self.template_name = template_name
|
|
self.custom_filters = custom_filters or []
|
|
|
|
# The name of the JavaScript function that will output this template. By using a named
|
|
# function the template can call itself which is required to support recursive includes.
|
|
self.js_function_name = "template" + "".join(
|
|
x.title()
|
|
for x in re.split(r"[^\w]|_", path.splitext(self.template_name)[0])
|
|
)
|
|
|
|
self.context_name = "context"
|
|
|
|
self._add_dependency(self.runtime_path, "jinjaToJS")
|
|
|
|
# Jinja2 doesn't accept Windows filepaths
|
|
if os.name == "nt":
|
|
self.template_name = self.template_name.replace(os.pathsep, "/")
|
|
|
|
template_string, template_path, _ = self.environment.loader.get_source(
|
|
self.environment, self.template_name
|
|
)
|
|
|
|
# It is assumed that this will be the absolute path to the template. It is used to work out
|
|
# related paths for inclues.
|
|
self.template_path = template_path
|
|
|
|
if self.js_module_format not in JS_MODULE_FORMATS.keys():
|
|
raise ValueError(
|
|
"The js_module_format option must be one of: %s"
|
|
% JS_MODULE_FORMATS.keys()
|
|
)
|
|
|
|
self.ast = self.environment.parse(template_string)
|
|
|
|
try:
|
|
for node in self.ast.body:
|
|
self._process_node(node)
|
|
except ExtendsException:
|
|
pass
|
|
|
|
def get_output(self):
|
|
"""
|
|
Returns the generated JavaScript code.
|
|
|
|
Returns:
|
|
str
|
|
"""
|
|
# generate the JS function string
|
|
template_function = TEMPLATE_WRAPPER.format(
|
|
function_name=self.js_function_name, template_code=self.output.getvalue()
|
|
).strip()
|
|
|
|
# get the correct module format template
|
|
module_format = JS_MODULE_FORMATS[self.js_module_format]
|
|
|
|
# generate the module code
|
|
return module_format(self.dependencies, template_function)
|
|
|
|
def _get_depencency_var_name(self, dependency):
|
|
"""
|
|
Returns the variable name assigned to the given dependency or None if the dependency has
|
|
not yet been registered.
|
|
|
|
Args:
|
|
dependency (str): Thet dependency that needs to be imported.
|
|
|
|
Returns:
|
|
str or None
|
|
"""
|
|
for dep_path, var_name in self.dependencies:
|
|
if dep_path == dependency:
|
|
return var_name
|
|
|
|
def _add_dependency(self, dependency, var_name=None):
|
|
"""
|
|
Adds the given dependency and returns the variable name to use to access it. If `var_name`
|
|
is not given then a random one will be created.
|
|
|
|
Args:
|
|
dependency (str):
|
|
var_name (str, optional):
|
|
|
|
Returns:
|
|
str
|
|
"""
|
|
if var_name is None:
|
|
var_name = next(self.temp_var_names)
|
|
# Don't add duplicate dependencies
|
|
if (dependency, var_name) not in self.dependencies:
|
|
self.dependencies.append((dependency, var_name))
|
|
return var_name
|
|
|
|
def _process_node(self, node, **kwargs):
|
|
node_name = node.__class__.__name__.lower()
|
|
handler = getattr(self, "_process_" + node_name, None)
|
|
if callable(handler):
|
|
handler(node, **kwargs)
|
|
else:
|
|
raise Exception(f"Unknown node {node} ({node_name})")
|
|
|
|
def _process_extends(self, node, **kwargs):
|
|
"""
|
|
Processes an extends block e.g. `{% extends "some/template.jinja" %}`
|
|
"""
|
|
|
|
# find all the blocks in this template
|
|
for b in self.ast.find_all(nodes.Block):
|
|
|
|
# if not already in `child_blocks` then this is the first time a
|
|
# block with this name has been encountered.
|
|
if b.name not in self.child_blocks:
|
|
self.child_blocks[b.name] = b
|
|
else:
|
|
|
|
# otherwise we have seen this block before, so we need to find the last
|
|
# super_block and add the block from this template to the end.
|
|
block = self.child_blocks.get(b.name)
|
|
while hasattr(block, "super_block"):
|
|
block = block.super_block
|
|
block.super_block = b
|
|
|
|
# load the parent template
|
|
parent_template = JinjaToJS(
|
|
template_root=self.template_root,
|
|
template_name=node.template.value,
|
|
js_module_format=self.js_module_format,
|
|
runtime_path=self.runtime_path,
|
|
include_prefix=self.include_prefix,
|
|
include_ext=self.include_ext,
|
|
child_blocks=self.child_blocks,
|
|
dependencies=self.dependencies,
|
|
)
|
|
|
|
# add the parent templates output to the current output
|
|
self.output.write(parent_template.output.getvalue())
|
|
|
|
# Raise an exception so we stop parsing this template
|
|
raise ExtendsException
|
|
|
|
def _process_block(self, node, **kwargs):
|
|
"""
|
|
Processes a block e.g. `{% block my_block %}{% endblock %}`
|
|
"""
|
|
|
|
# check if this node already has a 'super_block' attribute
|
|
if not hasattr(node, "super_block"):
|
|
|
|
# since it doesn't it must be the last block in the inheritance chain
|
|
node.super_block = None
|
|
|
|
# see if there has been a child block defined - if there is this
|
|
# will be the first block in the inheritance chain
|
|
child_block = self.child_blocks.get(node.name)
|
|
|
|
if child_block:
|
|
|
|
# we have child nodes so we need to set `node` as the
|
|
# super of the last one in the chain
|
|
last_block = child_block
|
|
while hasattr(last_block, "super_block"):
|
|
last_block = child_block.super_block
|
|
|
|
# once we have found it, set this node as it's super block
|
|
last_block.super_block = node
|
|
|
|
# this is the node we want to process as it's the first in the inheritance chain
|
|
node = child_block
|
|
|
|
# process the block passing the it's super along, if this block
|
|
# calls super() it will be handled by `_process_call`
|
|
for n in node.body:
|
|
self._process_node(n, super_block=node.super_block, **kwargs)
|
|
|
|
def _process_output(self, node, **kwargs):
|
|
"""
|
|
Processes an output node, which will contain things like `Name` and `TemplateData` nodes.
|
|
"""
|
|
for n in node.nodes:
|
|
self._process_node(n, **kwargs)
|
|
|
|
def _process_templatedata(self, node, **_):
|
|
"""
|
|
Processes a `TemplateData` node, this is just a bit of as-is text
|
|
to be written to the output.
|
|
"""
|
|
|
|
# escape double quotes
|
|
value = re.sub('"', r'\\"', node.data)
|
|
|
|
# escape new lines
|
|
value = re.sub("\n", r"\\n", value)
|
|
|
|
# append value to the result
|
|
self.output.write('__result += "' + value + '";')
|
|
|
|
def _process_name(self, node, **kwargs):
|
|
"""
|
|
Processes a `Name` node. Some examples of `Name` nodes:
|
|
{{ foo }} -> 'foo' is a Name
|
|
{% if foo }} -> 'foo' is a Name
|
|
"""
|
|
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs):
|
|
|
|
if node.name not in self.stored_names and node.ctx != "store":
|
|
self.output.write(self.context_name)
|
|
self.output.write(".")
|
|
|
|
if node.ctx == "store":
|
|
self.stored_names.add(node.name)
|
|
|
|
self.output.write(node.name)
|
|
|
|
def _process_dict(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs):
|
|
if node.items:
|
|
err = f"Can't process non-empty dict in expression: {node}"
|
|
raise ValueError(err)
|
|
self.output.write("{}")
|
|
|
|
def _process_getattr(self, node, **kwargs):
|
|
"""
|
|
Processes a `GetAttr` node. e.g. {{ foo.bar }}
|
|
"""
|
|
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
if is_loop_helper(node):
|
|
self._process_loop_helper(node, **new_kwargs)
|
|
else:
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(".")
|
|
self.output.write(node.attr)
|
|
|
|
def _process_getitem(self, node, **kwargs):
|
|
"""
|
|
Processes a `GetItem` node e.g. {{ foo["bar"] }}
|
|
"""
|
|
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
|
|
if isinstance(node.arg, nodes.Slice):
|
|
self.output.write(".slice(")
|
|
|
|
if node.arg.step is not None:
|
|
raise Exception(
|
|
"The step argument is not supported when slicing."
|
|
)
|
|
|
|
if node.arg.start is None:
|
|
self.output.write("0")
|
|
else:
|
|
self._process_node(node.arg.start, **new_kwargs)
|
|
|
|
if node.arg.stop is None:
|
|
self.output.write(")")
|
|
else:
|
|
self.output.write(",")
|
|
self._process_node(node.arg.stop, **new_kwargs)
|
|
self.output.write(")")
|
|
else:
|
|
self.output.write("[")
|
|
self._process_node(node.arg, **new_kwargs)
|
|
self.output.write("]")
|
|
|
|
def _process_for(self, node, **kwargs):
|
|
"""
|
|
Processes a for loop. e.g.
|
|
{% for number in numbers %}
|
|
{{ number }}
|
|
{% endfor %}
|
|
{% for key, value in somemap.items() %}
|
|
{{ key }} -> {{ value }}
|
|
{% %}
|
|
"""
|
|
|
|
# since a for loop can introduce new names into the context
|
|
# we need to remember the ones that existed outside the loop
|
|
previous_stored_names = self.stored_names.copy()
|
|
|
|
with self._execution():
|
|
self.output.write("__runtime.each(")
|
|
|
|
if is_method_call(node.iter, dict.keys.__name__):
|
|
self.output.write("Object.keys(")
|
|
|
|
self._process_node(node.iter, **kwargs)
|
|
|
|
if is_method_call(node.iter, dict.keys.__name__):
|
|
self.output.write(")")
|
|
|
|
self.output.write(",")
|
|
self.output.write("function")
|
|
self.output.write("(")
|
|
|
|
# javascript iterations put the value first, then the key
|
|
if isinstance(node.target, nodes.Tuple):
|
|
if len(node.target.items) > 2:
|
|
raise Exception(
|
|
"De-structuring more than 2 items is not supported."
|
|
)
|
|
|
|
for i, item in enumerate(reversed(node.target.items)):
|
|
self._process_node(item, **kwargs)
|
|
if i < len(node.target.items) - 1:
|
|
self.output.write(",")
|
|
else:
|
|
self._process_node(node.target, **kwargs)
|
|
|
|
self.output.write(")")
|
|
self.output.write("{")
|
|
|
|
if node.test:
|
|
self.output.write("if (!(")
|
|
self._process_node(node.test, **kwargs)
|
|
self.output.write(")) { return; }")
|
|
|
|
assigns = (
|
|
node.target.items if isinstance(node.target, nodes.Tuple) else [node.target]
|
|
)
|
|
|
|
with self._scoped_variables(assigns, **kwargs):
|
|
for n in node.body:
|
|
self._process_node(n, **kwargs)
|
|
|
|
with self._execution():
|
|
self.output.write("}")
|
|
self.output.write(")")
|
|
self.output.write(";")
|
|
|
|
# restore the stored names
|
|
self.stored_names = previous_stored_names
|
|
|
|
def _process_if(self, node, execute_end=None, **kwargs):
|
|
"""
|
|
Processes an if block e.g. `{% if foo %} do something {% endif %}`
|
|
"""
|
|
|
|
with self._execution():
|
|
self.output.write("if")
|
|
self.output.write("(")
|
|
|
|
with option(kwargs, use_python_bool_wrapper=True):
|
|
self._process_node(node.test, **kwargs)
|
|
|
|
self.output.write(")")
|
|
self.output.write("{")
|
|
|
|
# We accept an `execute_end` function as a keyword argument as this function is
|
|
# recursive in the case of something like if-elif-elif-else. In these cases this
|
|
# invocation of this function may have to close execution opened by a previous
|
|
# invocation of this function.
|
|
if execute_end:
|
|
execute_end()
|
|
|
|
# body
|
|
for n in node.body:
|
|
self._process_node(n, **kwargs)
|
|
|
|
if not node.else_ and not node.elif_:
|
|
# no else - just close the if
|
|
with self._execution():
|
|
self.output.write("}")
|
|
|
|
else:
|
|
# either an else or an elif
|
|
with self._execution() as execute_end:
|
|
self.output.write("}")
|
|
self.output.write(" else ")
|
|
|
|
# check for elif
|
|
for n in node.elif_:
|
|
self._process_node(n, execute_end=execute_end, **kwargs)
|
|
|
|
if node.elif_ and node.else_:
|
|
self.output.write(" else ")
|
|
|
|
# open up the body
|
|
self.output.write("{")
|
|
|
|
# process the body of the else
|
|
for n in node.else_:
|
|
self._process_node(n, **kwargs)
|
|
|
|
# close the body
|
|
with self._execution():
|
|
self.output.write("}")
|
|
|
|
def _process_condexpr(self, node, **kwargs):
|
|
with self._interpolation():
|
|
self.output.write("(")
|
|
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.test, **new_kwargs)
|
|
|
|
self.output.write(" ? ")
|
|
self._process_node(node.expr1, **kwargs)
|
|
self.output.write(" : ")
|
|
self._process_node(node.expr2, **kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_not(self, node, **kwargs):
|
|
self.output.write("!")
|
|
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
|
|
def _process_or(self, node, **kwargs):
|
|
self._process_node(node.left, **kwargs)
|
|
self.output.write(" || ")
|
|
self._process_node(node.right, **kwargs)
|
|
|
|
def _process_and(self, node, **kwargs):
|
|
self._process_node(node.left, **kwargs)
|
|
self.output.write(" && ")
|
|
self._process_node(node.right, **kwargs)
|
|
|
|
def _process_tuple(self, node, **kwargs):
|
|
self.output.write("[")
|
|
for i, item in enumerate(node.items):
|
|
self._process_node(item, **kwargs)
|
|
if i < len(node.items) - 1:
|
|
self.output.write(",")
|
|
self.output.write("]")
|
|
|
|
def _process_call(self, node, super_block=None, **kwargs):
|
|
if is_method_call(node, DICT_ITER_METHODS):
|
|
# special case for dict methods
|
|
self._process_node(node.node.node, **kwargs)
|
|
|
|
elif is_method_call(node, "super"):
|
|
# special case for the super() method which is available inside blocks
|
|
if not super_block:
|
|
raise Exception("super() called outside of a block with a parent.")
|
|
self._process_node(super_block, **kwargs)
|
|
|
|
else:
|
|
# just a normal function call on a context variable
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write("(")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
# only output the semi-colon if we are not interpolating
|
|
if self.state != STATE_INTERPOLATING:
|
|
self.output.write("")
|
|
|
|
def _process_filter(self, node, **kwargs):
|
|
method_name = getattr(self, "_process_filter_%s" % node.name, None)
|
|
if callable(method_name):
|
|
method_name(node, **kwargs)
|
|
elif node.name in self.custom_filters:
|
|
with self._interpolation(safe=True):
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.%s(" % node.name)
|
|
self._process_node(node.node, **new_kwargs)
|
|
if getattr(node, "args", None):
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
else:
|
|
raise Exception("Unsupported filter: %s" % node.name)
|
|
|
|
def _process_filter_safe(self, node, **kwargs):
|
|
with self._interpolation(safe=True):
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
|
|
def _process_filter_capitalize(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.capitalize(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_abs(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("Math.abs(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_replace(self, node, **kwargs):
|
|
# We're getting a quoted string from Python/Jinja as the pattern to
|
|
# replace, but to replace all occurrences in JS, we typically need a
|
|
# regex, which would be annoying to convert. So we're using split/join
|
|
# instead here.
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(".split(")
|
|
self._process_node(node.args[0], **new_kwargs)
|
|
self.output.write(").join(")
|
|
self._process_node(node.args[1], **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_pprint(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("JSON.stringify(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_attr(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write("[")
|
|
self._process_node(node.args[0], **new_kwargs)
|
|
self.output.write("]")
|
|
|
|
def _process_filter_batch(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.batch(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_default(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.default(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
if node.args:
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_first(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.first(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_int(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.int(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
if node.args:
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_round(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("Math.round((")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write("+ Number.EPSILON) * 10**")
|
|
self._process_node(node.args[0], **new_kwargs)
|
|
self.output.write(") / 10**")
|
|
self._process_node(node.args[0], **new_kwargs)
|
|
|
|
def _process_filter_last(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.last(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_length(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.size(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_lower(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(' + "").toLowerCase()')
|
|
|
|
def _process_filter_slice(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.slice(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_title(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.title(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_filter_trim(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(' + "").trim()')
|
|
|
|
def _process_filter_upper(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(' + "").toUpperCase()')
|
|
|
|
def _process_filter_truncate(self, node, **kwargs):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self.output.write("__filters.truncate(")
|
|
self._process_node(node.node, **new_kwargs)
|
|
self.output.write(",")
|
|
self._process_args(node, **new_kwargs)
|
|
self.output.write(")")
|
|
|
|
def _process_assign(self, node, **kwargs):
|
|
with self._execution():
|
|
self.output.write("var ")
|
|
self._process_node(node.target, **kwargs)
|
|
self.output.write(" = ")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(";")
|
|
|
|
def _process_with(self, node, **kwargs):
|
|
|
|
# keep a copy of the stored names before the scope
|
|
previous_stored_names = self.stored_names.copy()
|
|
|
|
# assigns in the with tag
|
|
# e.g. {% with var = "something %}
|
|
assigns_in_tag = [nodes.Assign(t, v) for t, v in zip(node.targets, node.values)]
|
|
|
|
# assigns in the with body
|
|
# e.g. {% set name = 'John' %}
|
|
assigns_in_body = [x for x in node.body if isinstance(x, nodes.Assign)]
|
|
|
|
# remove assigns from the body
|
|
node.body = [x for x in node.body if not isinstance(x, nodes.Assign)]
|
|
|
|
# get a list of all the assigns in this with block
|
|
# both on the tag, and within the body of the block
|
|
all_assigns = assigns_in_tag + assigns_in_body
|
|
|
|
with self._execution():
|
|
self.output.write("(function () {")
|
|
|
|
with self._scoped_variables(all_assigns, **kwargs):
|
|
for node in node.body:
|
|
self._process_node(node, **kwargs)
|
|
|
|
with self._execution():
|
|
self.output.write("})();")
|
|
|
|
# restore previous stored names
|
|
self.stored_names = previous_stored_names
|
|
|
|
def _process_compare(self, node, **kwargs):
|
|
|
|
if len(node.ops) > 1:
|
|
raise Exception("Multiple operands are not supported.")
|
|
|
|
operand = node.ops[0]
|
|
is_equality = operand.op in ("eq", "ne")
|
|
left_hand_is_const = isinstance(node.expr, nodes.Const)
|
|
right_hand_is_const = isinstance(operand.expr, nodes.Const)
|
|
|
|
# If the operand is equality and neither the left or right hand side are constants then we
|
|
# will need to use the JavaScript deep equals function. Ideally we want to avoid using this
|
|
# as it is quite a big function.
|
|
use_is_equal_function = is_equality and not (
|
|
left_hand_is_const or right_hand_is_const
|
|
)
|
|
|
|
with option(kwargs, use_python_bool_wrapper=False):
|
|
if operand.op == "in" or operand.op == "notin":
|
|
# Special case for "in" operator
|
|
if operand.op == "notin":
|
|
self.output.write("!")
|
|
self._process_node(operand.expr, **kwargs)
|
|
self.output.write(".includes(")
|
|
self._process_node(node.expr, **kwargs)
|
|
self.output.write(")")
|
|
else:
|
|
if use_is_equal_function:
|
|
if operand.op == "ne":
|
|
self.output.write("!")
|
|
self.output.write("__runtime.isEqual(")
|
|
|
|
self._process_node(node.expr, **kwargs)
|
|
|
|
if use_is_equal_function:
|
|
self.output.write(",")
|
|
else:
|
|
self.output.write(OPERANDS.get(operand.op))
|
|
|
|
self._process_node(operand.expr, **kwargs)
|
|
|
|
if use_is_equal_function:
|
|
self.output.write(")")
|
|
|
|
def _process_operand(self, node, **kwargs):
|
|
self.output.write(OPERANDS.get(node.op))
|
|
self._process_node(node.expr, **kwargs)
|
|
|
|
def _process_const(self, node, **_):
|
|
with self._interpolation():
|
|
self.output.write(json.dumps(node.value))
|
|
|
|
def _process_nonetype(self, node, **_):
|
|
with self._interpolation():
|
|
self.output.write("null")
|
|
|
|
def _process_neg(self, node, **kwargs):
|
|
with self._interpolation():
|
|
self.output.write("-")
|
|
self._process_node(node.node, **kwargs)
|
|
|
|
def _process_list(self, node, **kwargs):
|
|
self.output.write("[")
|
|
for i, item in enumerate(node.items):
|
|
self._process_node(item, **kwargs)
|
|
if i < len(node.items) - 1:
|
|
self.output.write(",")
|
|
self.output.write("]")
|
|
|
|
def _process_test(self, node, **kwargs):
|
|
with option(kwargs, use_python_bool_wrapper=False):
|
|
method_name = getattr(self, "_process_test_%s" % node.name, None)
|
|
if callable(method_name):
|
|
method_name(node, **kwargs)
|
|
else:
|
|
raise Exception("Unsupported test: %s" % node.name)
|
|
|
|
def _process_test_defined(self, node, **kwargs):
|
|
self.output.write("(typeof ")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(' !== "undefined")')
|
|
|
|
def _process_test_undefined(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(" === undefined")
|
|
|
|
def _process_test_callable(self, node, **kwargs):
|
|
self.output.write("__runtime.type(")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(') === "Function"')
|
|
|
|
def _process_test_divisibleby(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(" % ")
|
|
self._process_node(node.args[0], **kwargs)
|
|
self.output.write(" === 0")
|
|
|
|
def _process_test_even(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(" % 2 === 0")
|
|
|
|
def _process_test_odd(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(" % 2 === 1")
|
|
|
|
def _process_test_none(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(" === null")
|
|
|
|
def _process_test_upper(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(".toUpperCase() === ")
|
|
self._process_node(node.node, **kwargs)
|
|
|
|
def _process_test_lower(self, node, **kwargs):
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(".toLowerCase() === ")
|
|
self._process_node(node.node, **kwargs)
|
|
|
|
def _process_test_string(self, node, **kwargs):
|
|
self.output.write("__runtime.type(")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(') === "String"')
|
|
|
|
def _process_test_mapping(self, node, **kwargs):
|
|
self.output.write("__runtime.type(")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(') === "Object"')
|
|
|
|
def _process_test_number(self, node, **kwargs):
|
|
self.output.write("(__runtime.type(")
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write(') === "Number" && !isNaN(')
|
|
self._process_node(node.node, **kwargs)
|
|
self.output.write("))")
|
|
|
|
def _process_include(self, node, **kwargs):
|
|
with self._interpolation(safe=True):
|
|
include_path = node.template.value
|
|
|
|
if include_path == self.template_name:
|
|
# template is including itself
|
|
include_var_name = self.js_function_name
|
|
else:
|
|
if self.include_prefix:
|
|
include_path = self.include_prefix + node.template.value
|
|
elif (
|
|
self.js_module_format in ("es6", "commonjs",) and self.template_name
|
|
):
|
|
_, absolute_include_path, _ = self.environment.loader.get_source(
|
|
self.environment, node.template.value
|
|
)
|
|
include_path = os.path.relpath(
|
|
absolute_include_path, os.path.dirname(self.template_path)
|
|
)
|
|
if not include_path.startswith("."):
|
|
include_path = "./" + include_path
|
|
|
|
# Jinja2 doesn't accept Windows filepaths (but does output them!)
|
|
if os.name == "nt":
|
|
include_path = include_path.replace(os.pathsep, "/")
|
|
|
|
include_path = path.splitext(include_path)[0] + self.include_ext
|
|
include_var_name = self._get_depencency_var_name(include_path)
|
|
|
|
if not include_var_name:
|
|
include_var_name = self._add_dependency(include_path)
|
|
|
|
if self.js_module_format is None:
|
|
self.output.write('jinjaToJS.include("')
|
|
self.output.write(include_path)
|
|
self.output.write('");')
|
|
else:
|
|
self.output.write(include_var_name)
|
|
|
|
self.output.write("(")
|
|
self.output.write(self.context_name)
|
|
self.output.write(")")
|
|
|
|
def _process_add(self, node, **kwargs):
|
|
# Handle + operator for lists, which behaves differently in JS. Currently
|
|
# only works if we have an explicit list node on either side (in which
|
|
# case we assume both are lists).
|
|
if isinstance(node.left, nodes.List) or isinstance(node.right, nodes.List):
|
|
with self._interpolation():
|
|
with self._python_bool_wrapper(**kwargs) as new_kwargs:
|
|
self._process_node(node.left, **new_kwargs)
|
|
self.output.write(".concat(")
|
|
self._process_node(node.right, **new_kwargs)
|
|
self.output.write(")")
|
|
else:
|
|
self._process_math(node, math_operator=" + ", **kwargs)
|
|
|
|
def _process_sub(self, node, **kwargs):
|
|
self._process_math(node, math_operator=" - ", **kwargs)
|
|
|
|
def _process_div(self, node, **kwargs):
|
|
self._process_math(node, math_operator=" / ", **kwargs)
|
|
|
|
def _process_floordiv(self, node, **kwargs):
|
|
self._process_math(node, math_operator=" / ", function="Math.floor", **kwargs)
|
|
|
|
def _process_mul(self, node, **kwargs):
|
|
self._process_math(node, math_operator=" * ", **kwargs)
|
|
|
|
def _process_mod(self, node, **kwargs):
|
|
self._process_math(node, math_operator=" % ", **kwargs)
|
|
|
|
def _process_math(self, node, math_operator=None, function=None, **kwargs):
|
|
"""
|
|
Processes a math node e.g. `Div`, `Sub`, `Add`, `Mul` etc...
|
|
If `function` is provided the expression is wrapped in a call to that function.
|
|
"""
|
|
|
|
with self._interpolation():
|
|
if function:
|
|
self.output.write(function)
|
|
self.output.write("(")
|
|
|
|
self._process_node(node.left, **kwargs)
|
|
self.output.write(math_operator)
|
|
self._process_node(node.right, **kwargs)
|
|
|
|
if function:
|
|
self.output.write(")")
|
|
|
|
def _process_loop_helper(self, node, **kwargs):
|
|
"""
|
|
Processes a loop helper e.g. {{ loop.first }} or {{ loop.index }}
|
|
"""
|
|
|
|
if node.attr == LOOP_HELPER_INDEX:
|
|
self.output.write("(arguments[1] + 1)")
|
|
elif node.attr == LOOP_HELPER_INDEX_0:
|
|
self.output.write("arguments[1]")
|
|
elif node.attr == LOOP_HELPER_FIRST:
|
|
self.output.write("(arguments[1] == 0)")
|
|
elif node.attr == LOOP_HELPER_LAST:
|
|
self.output.write("(arguments[1] == arguments[2].length - 1)")
|
|
elif node.attr == LOOP_HELPER_LENGTH:
|
|
self.output.write("arguments[2].length")
|
|
|
|
def _process_args(self, node, **kwargs):
|
|
args = getattr(node, "args", None)
|
|
if not args:
|
|
return
|
|
for i, item in enumerate(args):
|
|
self._process_node(item, **kwargs)
|
|
if i < len(node.args) - 1:
|
|
self.output.write(",")
|
|
|
|
@contextlib.contextmanager
|
|
def _execution(self):
|
|
"""
|
|
Context manager for executing some JavaScript inside a template.
|
|
"""
|
|
|
|
did_start_executing = False
|
|
|
|
if self.state == STATE_DEFAULT:
|
|
did_start_executing = True
|
|
self.state = STATE_EXECUTING
|
|
|
|
def close():
|
|
if did_start_executing and self.state == STATE_EXECUTING:
|
|
self.state = STATE_DEFAULT
|
|
|
|
yield close
|
|
close()
|
|
|
|
@contextlib.contextmanager
|
|
def _interpolation(self, safe=False):
|
|
|
|
did_start_interpolating = False
|
|
|
|
if self.state == STATE_DEFAULT:
|
|
did_start_interpolating = True
|
|
self.output.write('__result += "" + ')
|
|
if safe is not True:
|
|
self.output.write("__runtime.escape")
|
|
self.output.write("((__tmp = (")
|
|
self.state = STATE_INTERPOLATING
|
|
|
|
def close():
|
|
if did_start_interpolating and self.state == STATE_INTERPOLATING:
|
|
self.output.write(')) == null ? "" : __tmp);')
|
|
self.state = STATE_DEFAULT
|
|
|
|
yield close
|
|
close()
|
|
|
|
@contextlib.contextmanager
|
|
def _scoped_variables(self, nodes_list, **kwargs):
|
|
"""
|
|
Context manager for creating scoped variables defined by the nodes in `nodes_list`.
|
|
These variables will be added to the context, and when the context manager exits the
|
|
context object will be restored to it's previous state.
|
|
"""
|
|
|
|
tmp_vars = []
|
|
for node in nodes_list:
|
|
|
|
is_assign_node = isinstance(node, nodes.Assign)
|
|
name = node.target.name if is_assign_node else node.name
|
|
|
|
# create a temp variable name
|
|
tmp_var = next(self.temp_var_names)
|
|
|
|
# save previous context value
|
|
with self._execution():
|
|
|
|
# save the current value of this name
|
|
self.output.write(
|
|
"var %s = %s.%s;" % (tmp_var, self.context_name, name)
|
|
)
|
|
|
|
# add new value to context
|
|
self.output.write("%s.%s = " % (self.context_name, name))
|
|
|
|
if is_assign_node:
|
|
self._process_node(node.node, **kwargs)
|
|
else:
|
|
self.output.write(node.name)
|
|
|
|
self.output.write(";")
|
|
|
|
tmp_vars.append((tmp_var, name))
|
|
|
|
yield
|
|
|
|
# restore context
|
|
for tmp_var, name in tmp_vars:
|
|
with self._execution():
|
|
self.output.write("%s.%s = %s;" % (self.context_name, name, tmp_var))
|
|
|
|
@contextlib.contextmanager
|
|
def _python_bool_wrapper(self, **kwargs):
|
|
|
|
use_python_bool_wrapper = kwargs.get("use_python_bool_wrapper")
|
|
|
|
if use_python_bool_wrapper:
|
|
self.output.write("__runtime.boolean(")
|
|
|
|
with option(kwargs, use_python_bool_wrapper=False):
|
|
yield kwargs
|
|
|
|
if use_python_bool_wrapper:
|
|
self.output.write(")")
|
|
|
|
|
|
def main(template_path, output=None, data_path=None):
|
|
"""Convert a jinja2 template to a JavaScript module.
|
|
|
|
template_path (Path): Path to .jijna file.
|
|
output (Optional[Path]): Path to output .js module (stdout if unset).
|
|
data_path (Optional[Path]): Optional JSON or YAML file with additional data
|
|
to be included in the JS module as the exported variable DATA.
|
|
"""
|
|
data = "{}"
|
|
if data_path is not None:
|
|
if data_path.suffix in (".yml", ".yaml"):
|
|
data = srsly.read_yaml(data_path)
|
|
else:
|
|
data = srsly.read_json(data_path)
|
|
data = srsly.json_dumps(data) # dump and load for compactness
|
|
template_path = Path(template_path)
|
|
tpl_file = template_path.parts[-1]
|
|
compiler = JinjaToJS(template_path.parent, tpl_file, js_module_format="es6")
|
|
header = f"// This file was auto-generated by {__file__} based on {tpl_file}"
|
|
data_str = f"export const DATA = {data}"
|
|
result = compiler.get_output()
|
|
if output is not None:
|
|
with output.open("w") as f:
|
|
f.write(f"{header}\n{result}\n{data_str}")
|
|
print(f"Updated {output.parts[-1]}")
|
|
else:
|
|
print(result)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
args = sys.argv[1:]
|
|
if not len(args):
|
|
raise ValueError("Need at least one argument: path to .jinja template")
|
|
template_path = Path(args[0])
|
|
output = Path(args[1]) if len(args) > 1 else None
|
|
data_path = Path(args[2]) if len(args) > 2 else None
|
|
main(template_path, output, data_path)
|