aboutsummaryrefslogtreecommitdiffstats
path: root/mako/_ast_util.py
diff options
context:
space:
mode:
Diffstat (limited to 'mako/_ast_util.py')
-rw-r--r--mako/_ast_util.py388
1 files changed, 215 insertions, 173 deletions
diff --git a/mako/_ast_util.py b/mako/_ast_util.py
index e2c7d24..be95bfb 100644
--- a/mako/_ast_util.py
+++ b/mako/_ast_util.py
@@ -30,47 +30,77 @@
:copyright: Copyright 2008 by Armin Ronacher.
:license: Python License.
"""
-from _ast import * # noqa
+from _ast import Add
+from _ast import And
+from _ast import AST
+from _ast import BitAnd
+from _ast import BitOr
+from _ast import BitXor
+from _ast import ClassDef
+from _ast import Div
+from _ast import Eq
+from _ast import Expression
+from _ast import FloorDiv
+from _ast import FunctionDef
+from _ast import Gt
+from _ast import GtE
+from _ast import If
+from _ast import In
+from _ast import Interactive
+from _ast import Invert
+from _ast import Is
+from _ast import IsNot
+from _ast import LShift
+from _ast import Lt
+from _ast import LtE
+from _ast import Mod
+from _ast import mod
+from _ast import Module
+from _ast import Mult
+from _ast import Name
+from _ast import Not
+from _ast import NotEq
+from _ast import NotIn
+from _ast import Or
+from _ast import PyCF_ONLY_AST
+from _ast import RShift
+from _ast import Str
+from _ast import Sub
+from _ast import UAdd
+from _ast import USub
+
from mako.compat import arg_stringname
-BOOLOP_SYMBOLS = {
- And: 'and',
- Or: 'or'
-}
+BOOLOP_SYMBOLS = {And: "and", Or: "or"}
BINOP_SYMBOLS = {
- Add: '+',
- Sub: '-',
- Mult: '*',
- Div: '/',
- FloorDiv: '//',
- Mod: '%',
- LShift: '<<',
- RShift: '>>',
- BitOr: '|',
- BitAnd: '&',
- BitXor: '^'
+ Add: "+",
+ Sub: "-",
+ Mult: "*",
+ Div: "/",
+ FloorDiv: "//",
+ Mod: "%",
+ LShift: "<<",
+ RShift: ">>",
+ BitOr: "|",
+ BitAnd: "&",
+ BitXor: "^",
}
CMPOP_SYMBOLS = {
- Eq: '==',
- Gt: '>',
- GtE: '>=',
- In: 'in',
- Is: 'is',
- IsNot: 'is not',
- Lt: '<',
- LtE: '<=',
- NotEq: '!=',
- NotIn: 'not in'
+ Eq: "==",
+ Gt: ">",
+ GtE: ">=",
+ In: "in",
+ Is: "is",
+ IsNot: "is not",
+ Lt: "<",
+ LtE: "<=",
+ NotEq: "!=",
+ NotIn: "not in",
}
-UNARYOP_SYMBOLS = {
- Invert: '~',
- Not: 'not',
- UAdd: '+',
- USub: '-'
-}
+UNARYOP_SYMBOLS = {Invert: "~", Not: "not", UAdd: "+", USub: "-"}
ALL_SYMBOLS = {}
ALL_SYMBOLS.update(BOOLOP_SYMBOLS)
@@ -79,12 +109,12 @@ ALL_SYMBOLS.update(CMPOP_SYMBOLS)
ALL_SYMBOLS.update(UNARYOP_SYMBOLS)
-def parse(expr, filename='<unknown>', mode='exec'):
+def parse(expr, filename="<unknown>", mode="exec"):
"""Parse an expression into an AST node."""
return compile(expr, filename, mode, PyCF_ONLY_AST)
-def to_source(node, indent_with=' ' * 4):
+def to_source(node, indent_with=" " * 4):
"""
This function can convert a node tree back into python sourcecode. This
is useful for debugging purposes, especially if you're dealing with custom
@@ -101,7 +131,7 @@ def to_source(node, indent_with=' ' * 4):
"""
generator = SourceGenerator(indent_with)
generator.visit(node)
- return ''.join(generator.result)
+ return "".join(generator.result)
def dump(node):
@@ -109,16 +139,21 @@ def dump(node):
A very verbose representation of the node passed. This is useful for
debugging purposes.
"""
+
def _format(node):
if isinstance(node, AST):
- return '%s(%s)' % (node.__class__.__name__,
- ', '.join('%s=%s' % (a, _format(b))
- for a, b in iter_fields(node)))
+ return "%s(%s)" % (
+ node.__class__.__name__,
+ ", ".join(
+ "%s=%s" % (a, _format(b)) for a, b in iter_fields(node)
+ ),
+ )
elif isinstance(node, list):
- return '[%s]' % ', '.join(_format(x) for x in node)
+ return "[%s]" % ", ".join(_format(x) for x in node)
return repr(node)
+
if not isinstance(node, AST):
- raise TypeError('expected AST, got %r' % node.__class__.__name__)
+ raise TypeError("expected AST, got %r" % node.__class__.__name__)
return _format(node)
@@ -127,9 +162,12 @@ def copy_location(new_node, old_node):
Copy the source location hint (`lineno` and `col_offset`) from the
old to the new node if possible and return the new one.
"""
- for attr in 'lineno', 'col_offset':
- if attr in old_node._attributes and attr in new_node._attributes \
- and hasattr(old_node, attr):
+ for attr in "lineno", "col_offset":
+ if (
+ attr in old_node._attributes
+ and attr in new_node._attributes
+ and hasattr(old_node, attr)
+ ):
setattr(new_node, attr, getattr(old_node, attr))
return new_node
@@ -146,19 +184,21 @@ def fix_missing_locations(node):
Unlike `copy_location` this works recursive and won't touch nodes that
already have a location information.
"""
+
def _fix(node, lineno, col_offset):
- if 'lineno' in node._attributes:
- if not hasattr(node, 'lineno'):
+ if "lineno" in node._attributes:
+ if not hasattr(node, "lineno"):
node.lineno = lineno
else:
lineno = node.lineno
- if 'col_offset' in node._attributes:
- if not hasattr(node, 'col_offset'):
+ if "col_offset" in node._attributes:
+ if not hasattr(node, "col_offset"):
node.col_offset = col_offset
else:
col_offset = node.col_offset
for child in iter_child_nodes(node):
_fix(child, lineno, col_offset)
+
_fix(node, 1, 0)
return node
@@ -170,14 +210,14 @@ def increment_lineno(node, n=1):
file.
"""
for node in zip((node,), walk(node)):
- if 'lineno' in node._attributes:
- node.lineno = getattr(node, 'lineno', 0) + n
+ if "lineno" in node._attributes:
+ node.lineno = getattr(node, "lineno", 0) + n
def iter_fields(node):
"""Iterate over all fields of a node, only yielding existing fields."""
# CPython 2.5 compat
- if not hasattr(node, '_fields') or not node._fields:
+ if not hasattr(node, "_fields") or not node._fields:
return
for field in node._fields:
try:
@@ -213,11 +253,10 @@ def get_compile_mode(node):
node (`Expression`, `Module` etc.) a `TypeError` is thrown.
"""
if not isinstance(node, mod):
- raise TypeError('expected mod node, got %r' % node.__class__.__name__)
- return {
- Expression: 'eval',
- Interactive: 'single'
- }.get(node.__class__, 'expr')
+ raise TypeError("expected mod node, got %r" % node.__class__.__name__)
+ return {Expression: "eval", Interactive: "single"}.get(
+ node.__class__, "expr"
+ )
def get_docstring(node):
@@ -238,6 +277,7 @@ def walk(node):
place and don't care about the context or the order the nodes are returned.
"""
from collections import deque
+
todo = deque([node])
while todo:
node = todo.popleft()
@@ -269,7 +309,7 @@ class NodeVisitor(object):
exists for this node. In that case the generic visit function is
used instead.
"""
- method = 'visit_' + node.__class__.__name__
+ method = "visit_" + node.__class__.__name__
return getattr(self, method, None)
def visit(self, node):
@@ -367,7 +407,7 @@ class SourceGenerator(NodeVisitor):
def write(self, x):
if self.new_lines:
if self.result:
- self.result.append('\n' * self.new_lines)
+ self.result.append("\n" * self.new_lines)
self.result.append(self.indent_with * self.indentation)
self.new_lines = 0
self.result.append(x)
@@ -386,7 +426,7 @@ class SourceGenerator(NodeVisitor):
self.body(node.body)
if node.orelse:
self.newline()
- self.write('else:')
+ self.write("else:")
self.body(node.orelse)
def signature(self, node):
@@ -394,7 +434,7 @@ class SourceGenerator(NodeVisitor):
def write_comma():
if want_comma:
- self.write(', ')
+ self.write(", ")
else:
want_comma.append(True)
@@ -403,19 +443,19 @@ class SourceGenerator(NodeVisitor):
write_comma()
self.visit(arg)
if default is not None:
- self.write('=')
+ self.write("=")
self.visit(default)
if node.vararg is not None:
write_comma()
- self.write('*' + arg_stringname(node.vararg))
+ self.write("*" + arg_stringname(node.vararg))
if node.kwarg is not None:
write_comma()
- self.write('**' + arg_stringname(node.kwarg))
+ self.write("**" + arg_stringname(node.kwarg))
def decorators(self, node):
for decorator in node.decorator_list:
self.newline()
- self.write('@')
+ self.write("@")
self.visit(decorator)
# Statements
@@ -424,29 +464,29 @@ class SourceGenerator(NodeVisitor):
self.newline()
for idx, target in enumerate(node.targets):
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(target)
- self.write(' = ')
+ self.write(" = ")
self.visit(node.value)
def visit_AugAssign(self, node):
self.newline()
self.visit(node.target)
- self.write(BINOP_SYMBOLS[type(node.op)] + '=')
+ self.write(BINOP_SYMBOLS[type(node.op)] + "=")
self.visit(node.value)
def visit_ImportFrom(self, node):
self.newline()
- self.write('from %s%s import ' % ('.' * node.level, node.module))
+ self.write("from %s%s import " % ("." * node.level, node.module))
for idx, item in enumerate(node.names):
if idx:
- self.write(', ')
+ self.write(", ")
self.write(item)
def visit_Import(self, node):
self.newline()
for item in node.names:
- self.write('import ')
+ self.write("import ")
self.visit(item)
def visit_Expr(self, node):
@@ -457,9 +497,9 @@ class SourceGenerator(NodeVisitor):
self.newline(n=2)
self.decorators(node)
self.newline()
- self.write('def %s(' % node.name)
+ self.write("def %s(" % node.name)
self.signature(node.args)
- self.write('):')
+ self.write("):")
self.body(node.body)
def visit_ClassDef(self, node):
@@ -467,200 +507,200 @@ class SourceGenerator(NodeVisitor):
def paren_or_comma():
if have_args:
- self.write(', ')
+ self.write(", ")
else:
have_args.append(True)
- self.write('(')
+ self.write("(")
self.newline(n=3)
self.decorators(node)
self.newline()
- self.write('class %s' % node.name)
+ self.write("class %s" % node.name)
for base in node.bases:
paren_or_comma()
self.visit(base)
# XXX: the if here is used to keep this module compatible
# with python 2.6.
- if hasattr(node, 'keywords'):
+ if hasattr(node, "keywords"):
for keyword in node.keywords:
paren_or_comma()
- self.write(keyword.arg + '=')
+ self.write(keyword.arg + "=")
self.visit(keyword.value)
if getattr(node, "starargs", None):
paren_or_comma()
- self.write('*')
+ self.write("*")
self.visit(node.starargs)
if getattr(node, "kwargs", None):
paren_or_comma()
- self.write('**')
+ self.write("**")
self.visit(node.kwargs)
- self.write(have_args and '):' or ':')
+ self.write(have_args and "):" or ":")
self.body(node.body)
def visit_If(self, node):
self.newline()
- self.write('if ')
+ self.write("if ")
self.visit(node.test)
- self.write(':')
+ self.write(":")
self.body(node.body)
while True:
else_ = node.orelse
if len(else_) == 1 and isinstance(else_[0], If):
node = else_[0]
self.newline()
- self.write('elif ')
+ self.write("elif ")
self.visit(node.test)
- self.write(':')
+ self.write(":")
self.body(node.body)
else:
self.newline()
- self.write('else:')
+ self.write("else:")
self.body(else_)
break
def visit_For(self, node):
self.newline()
- self.write('for ')
+ self.write("for ")
self.visit(node.target)
- self.write(' in ')
+ self.write(" in ")
self.visit(node.iter)
- self.write(':')
+ self.write(":")
self.body_or_else(node)
def visit_While(self, node):
self.newline()
- self.write('while ')
+ self.write("while ")
self.visit(node.test)
- self.write(':')
+ self.write(":")
self.body_or_else(node)
def visit_With(self, node):
self.newline()
- self.write('with ')
+ self.write("with ")
self.visit(node.context_expr)
if node.optional_vars is not None:
- self.write(' as ')
+ self.write(" as ")
self.visit(node.optional_vars)
- self.write(':')
+ self.write(":")
self.body(node.body)
def visit_Pass(self, node):
self.newline()
- self.write('pass')
+ self.write("pass")
def visit_Print(self, node):
# XXX: python 2.6 only
self.newline()
- self.write('print ')
+ self.write("print ")
want_comma = False
if node.dest is not None:
- self.write(' >> ')
+ self.write(" >> ")
self.visit(node.dest)
want_comma = True
for value in node.values:
if want_comma:
- self.write(', ')
+ self.write(", ")
self.visit(value)
want_comma = True
if not node.nl:
- self.write(',')
+ self.write(",")
def visit_Delete(self, node):
self.newline()
- self.write('del ')
+ self.write("del ")
for idx, target in enumerate(node):
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(target)
def visit_TryExcept(self, node):
self.newline()
- self.write('try:')
+ self.write("try:")
self.body(node.body)
for handler in node.handlers:
self.visit(handler)
def visit_TryFinally(self, node):
self.newline()
- self.write('try:')
+ self.write("try:")
self.body(node.body)
self.newline()
- self.write('finally:')
+ self.write("finally:")
self.body(node.finalbody)
def visit_Global(self, node):
self.newline()
- self.write('global ' + ', '.join(node.names))
+ self.write("global " + ", ".join(node.names))
def visit_Nonlocal(self, node):
self.newline()
- self.write('nonlocal ' + ', '.join(node.names))
+ self.write("nonlocal " + ", ".join(node.names))
def visit_Return(self, node):
self.newline()
- self.write('return ')
+ self.write("return ")
self.visit(node.value)
def visit_Break(self, node):
self.newline()
- self.write('break')
+ self.write("break")
def visit_Continue(self, node):
self.newline()
- self.write('continue')
+ self.write("continue")
def visit_Raise(self, node):
# XXX: Python 2.6 / 3.0 compatibility
self.newline()
- self.write('raise')
- if hasattr(node, 'exc') and node.exc is not None:
- self.write(' ')
+ self.write("raise")
+ if hasattr(node, "exc") and node.exc is not None:
+ self.write(" ")
self.visit(node.exc)
if node.cause is not None:
- self.write(' from ')
+ self.write(" from ")
self.visit(node.cause)
- elif hasattr(node, 'type') and node.type is not None:
+ elif hasattr(node, "type") and node.type is not None:
self.visit(node.type)
if node.inst is not None:
- self.write(', ')
+ self.write(", ")
self.visit(node.inst)
if node.tback is not None:
- self.write(', ')
+ self.write(", ")
self.visit(node.tback)
# Expressions
def visit_Attribute(self, node):
self.visit(node.value)
- self.write('.' + node.attr)
+ self.write("." + node.attr)
def visit_Call(self, node):
want_comma = []
def write_comma():
if want_comma:
- self.write(', ')
+ self.write(", ")
else:
want_comma.append(True)
self.visit(node.func)
- self.write('(')
+ self.write("(")
for arg in node.args:
write_comma()
self.visit(arg)
for keyword in node.keywords:
write_comma()
- self.write(keyword.arg + '=')
+ self.write(keyword.arg + "=")
self.visit(keyword.value)
if getattr(node, "starargs", None):
write_comma()
- self.write('*')
+ self.write("*")
self.visit(node.starargs)
if getattr(node, "kwargs", None):
write_comma()
- self.write('**')
+ self.write("**")
self.visit(node.kwargs)
- self.write(')')
+ self.write(")")
def visit_Name(self, node):
self.write(node.id)
@@ -685,105 +725,106 @@ class SourceGenerator(NodeVisitor):
self.write(repr(node.value))
def visit_Tuple(self, node):
- self.write('(')
+ self.write("(")
idx = -1
for idx, item in enumerate(node.elts):
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(item)
- self.write(idx and ')' or ',)')
+ self.write(idx and ")" or ",)")
def sequence_visit(left, right):
def visit(self, node):
self.write(left)
for idx, item in enumerate(node.elts):
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(item)
self.write(right)
+
return visit
- visit_List = sequence_visit('[', ']')
- visit_Set = sequence_visit('{', '}')
+ visit_List = sequence_visit("[", "]")
+ visit_Set = sequence_visit("{", "}")
del sequence_visit
def visit_Dict(self, node):
- self.write('{')
+ self.write("{")
for idx, (key, value) in enumerate(zip(node.keys, node.values)):
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(key)
- self.write(': ')
+ self.write(": ")
self.visit(value)
- self.write('}')
+ self.write("}")
def visit_BinOp(self, node):
- self.write('(')
+ self.write("(")
self.visit(node.left)
- self.write(' %s ' % BINOP_SYMBOLS[type(node.op)])
+ self.write(" %s " % BINOP_SYMBOLS[type(node.op)])
self.visit(node.right)
- self.write(')')
+ self.write(")")
def visit_BoolOp(self, node):
- self.write('(')
+ self.write("(")
for idx, value in enumerate(node.values):
if idx:
- self.write(' %s ' % BOOLOP_SYMBOLS[type(node.op)])
+ self.write(" %s " % BOOLOP_SYMBOLS[type(node.op)])
self.visit(value)
- self.write(')')
+ self.write(")")
def visit_Compare(self, node):
- self.write('(')
+ self.write("(")
self.visit(node.left)
for op, right in zip(node.ops, node.comparators):
- self.write(' %s ' % CMPOP_SYMBOLS[type(op)])
+ self.write(" %s " % CMPOP_SYMBOLS[type(op)])
self.visit(right)
- self.write(')')
+ self.write(")")
def visit_UnaryOp(self, node):
- self.write('(')
+ self.write("(")
op = UNARYOP_SYMBOLS[type(node.op)]
self.write(op)
- if op == 'not':
- self.write(' ')
+ if op == "not":
+ self.write(" ")
self.visit(node.operand)
- self.write(')')
+ self.write(")")
def visit_Subscript(self, node):
self.visit(node.value)
- self.write('[')
+ self.write("[")
self.visit(node.slice)
- self.write(']')
+ self.write("]")
def visit_Slice(self, node):
if node.lower is not None:
self.visit(node.lower)
- self.write(':')
+ self.write(":")
if node.upper is not None:
self.visit(node.upper)
if node.step is not None:
- self.write(':')
- if not (isinstance(node.step, Name) and node.step.id == 'None'):
+ self.write(":")
+ if not (isinstance(node.step, Name) and node.step.id == "None"):
self.visit(node.step)
def visit_ExtSlice(self, node):
for idx, item in node.dims:
if idx:
- self.write(', ')
+ self.write(", ")
self.visit(item)
def visit_Yield(self, node):
- self.write('yield ')
+ self.write("yield ")
self.visit(node.value)
def visit_Lambda(self, node):
- self.write('lambda ')
+ self.write("lambda ")
self.signature(node.args)
- self.write(': ')
+ self.write(": ")
self.visit(node.body)
def visit_Ellipsis(self, node):
- self.write('Ellipsis')
+ self.write("Ellipsis")
def generator_visit(left, right):
def visit(self, node):
@@ -792,64 +833,65 @@ class SourceGenerator(NodeVisitor):
for comprehension in node.generators:
self.visit(comprehension)
self.write(right)
+
return visit
- visit_ListComp = generator_visit('[', ']')
- visit_GeneratorExp = generator_visit('(', ')')
- visit_SetComp = generator_visit('{', '}')
+ visit_ListComp = generator_visit("[", "]")
+ visit_GeneratorExp = generator_visit("(", ")")
+ visit_SetComp = generator_visit("{", "}")
del generator_visit
def visit_DictComp(self, node):
- self.write('{')
+ self.write("{")
self.visit(node.key)
- self.write(': ')
+ self.write(": ")
self.visit(node.value)
for comprehension in node.generators:
self.visit(comprehension)
- self.write('}')
+ self.write("}")
def visit_IfExp(self, node):
self.visit(node.body)
- self.write(' if ')
+ self.write(" if ")
self.visit(node.test)
- self.write(' else ')
+ self.write(" else ")
self.visit(node.orelse)
def visit_Starred(self, node):
- self.write('*')
+ self.write("*")
self.visit(node.value)
def visit_Repr(self, node):
# XXX: python 2.6 only
- self.write('`')
+ self.write("`")
self.visit(node.value)
- self.write('`')
+ self.write("`")
# Helper Nodes
def visit_alias(self, node):
self.write(node.name)
if node.asname is not None:
- self.write(' as ' + node.asname)
+ self.write(" as " + node.asname)
def visit_comprehension(self, node):
- self.write(' for ')
+ self.write(" for ")
self.visit(node.target)
- self.write(' in ')
+ self.write(" in ")
self.visit(node.iter)
if node.ifs:
for if_ in node.ifs:
- self.write(' if ')
+ self.write(" if ")
self.visit(if_)
def visit_excepthandler(self, node):
self.newline()
- self.write('except')
+ self.write("except")
if node.type is not None:
- self.write(' ')
+ self.write(" ")
self.visit(node.type)
if node.name is not None:
- self.write(' as ')
+ self.write(" as ")
self.visit(node.name)
- self.write(':')
+ self.write(":")
self.body(node.body)