diff options
| author | floatdrop <[email protected]> | 2022-02-10 16:47:15 +0300 | 
|---|---|---|
| committer | Daniil Cherednik <[email protected]> | 2022-02-10 16:47:15 +0300 | 
| commit | e63b84f1d39557d9e46ac380b1f388271894293c (patch) | |
| tree | 338cdaff3fb027e030b847db66df06019a0e3149 /contrib/python/Jinja2/py3/jinja2/nodes.py | |
| parent | f60febb7ea449535e7b073c386c7ff0539637fc0 (diff) | |
Restoring authorship annotation for <[email protected]>. Commit 1 of 2.
Diffstat (limited to 'contrib/python/Jinja2/py3/jinja2/nodes.py')
| -rw-r--r-- | contrib/python/Jinja2/py3/jinja2/nodes.py | 1418 | 
1 files changed, 709 insertions, 709 deletions
| diff --git a/contrib/python/Jinja2/py3/jinja2/nodes.py b/contrib/python/Jinja2/py3/jinja2/nodes.py index b2f88d9d9c1..8849eb12061 100644 --- a/contrib/python/Jinja2/py3/jinja2/nodes.py +++ b/contrib/python/Jinja2/py3/jinja2/nodes.py @@ -1,16 +1,16 @@  """AST nodes generated by the parser for the compiler. Also provides  some node tree helper functions used by the parser and compiler in order  to normalize nodes. -""" +"""   import inspect -import operator +import operator   import typing as t -from collections import deque - +from collections import deque  +   from markupsafe import Markup - +   from .utils import _PassArg - +   if t.TYPE_CHECKING:      import typing_extensions as te      from .environment import Environment @@ -25,14 +25,14 @@ _binop_to_func: t.Dict[str, t.Callable[[t.Any, t.Any], t.Any]] = {      "%": operator.mod,      "+": operator.add,      "-": operator.sub, -} - +}  +   _uaop_to_func: t.Dict[str, t.Callable[[t.Any], t.Any]] = {      "not": operator.not_,      "+": operator.pos,      "-": operator.neg,  } - +   _cmpop_to_func: t.Dict[str, t.Callable[[t.Any, t.Any], t.Any]] = {      "eq": operator.eq,      "ne": operator.ne, @@ -42,287 +42,287 @@ _cmpop_to_func: t.Dict[str, t.Callable[[t.Any, t.Any], t.Any]] = {      "lteq": operator.le,      "in": lambda a, b: a in b,      "notin": lambda a, b: a not in b, -} - - -class Impossible(Exception): -    """Raised if the node could not perform a requested action.""" - - -class NodeType(type): -    """A metaclass for nodes that handles the field and attribute -    inheritance.  fields and attributes from the parent class are -    automatically forwarded to the child.""" - +}  +  +  +class Impossible(Exception):  +    """Raised if the node could not perform a requested action."""  +  +  +class NodeType(type):  +    """A metaclass for nodes that handles the field and attribute  +    inheritance.  fields and attributes from the parent class are  +    automatically forwarded to the child."""  +       def __new__(mcs, name, bases, d):  # type: ignore          for attr in "fields", "attributes": -            storage = [] +            storage = []               storage.extend(getattr(bases[0] if bases else object, attr, ())) -            storage.extend(d.get(attr, ())) +            storage.extend(d.get(attr, ()))               assert len(bases) <= 1, "multiple inheritance not allowed"              assert len(storage) == len(set(storage)), "layout conflict" -            d[attr] = tuple(storage) +            d[attr] = tuple(storage)           d.setdefault("abstract", False)          return type.__new__(mcs, name, bases, d) - - +  +   class EvalContext: -    """Holds evaluation time information.  Custom attributes can be attached -    to it in extensions. -    """ - +    """Holds evaluation time information.  Custom attributes can be attached  +    to it in extensions.  +    """  +       def __init__(          self, environment: "Environment", template_name: t.Optional[str] = None      ) -> None: -        self.environment = environment -        if callable(environment.autoescape): -            self.autoescape = environment.autoescape(template_name) -        else: -            self.autoescape = environment.autoescape -        self.volatile = False - +        self.environment = environment  +        if callable(environment.autoescape):  +            self.autoescape = environment.autoescape(template_name)  +        else:  +            self.autoescape = environment.autoescape  +        self.volatile = False  +       def save(self) -> t.Mapping[str, t.Any]: -        return self.__dict__.copy() - +        return self.__dict__.copy()  +       def revert(self, old: t.Mapping[str, t.Any]) -> None: -        self.__dict__.clear() -        self.__dict__.update(old) - - +        self.__dict__.clear()  +        self.__dict__.update(old)  +  +   def get_eval_context(node: "Node", ctx: t.Optional[EvalContext]) -> EvalContext: -    if ctx is None: -        if node.environment is None: +    if ctx is None:  +        if node.environment is None:               raise RuntimeError(                  "if no eval context is passed, the node must have an"                  " attached environment."              ) -        return EvalContext(node.environment) -    return ctx - - +        return EvalContext(node.environment)  +    return ctx  +  +   class Node(metaclass=NodeType):      """Baseclass for all Jinja nodes.  There are a number of nodes available -    of different types.  There are four major types: - -    -   :class:`Stmt`: statements -    -   :class:`Expr`: expressions -    -   :class:`Helper`: helper nodes -    -   :class:`Template`: the outermost wrapper node - -    All nodes have fields and attributes.  Fields may be other nodes, lists, -    or arbitrary values.  Fields are passed to the constructor as regular -    positional arguments, attributes as keyword arguments.  Each node has -    two attributes: `lineno` (the line number of the node) and `environment`. -    The `environment` attribute is set at the end of the parsing process for -    all nodes automatically. -    """ +    of different types.  There are four major types:  +  +    -   :class:`Stmt`: statements  +    -   :class:`Expr`: expressions  +    -   :class:`Helper`: helper nodes  +    -   :class:`Template`: the outermost wrapper node  +  +    All nodes have fields and attributes.  Fields may be other nodes, lists,  +    or arbitrary values.  Fields are passed to the constructor as regular  +    positional arguments, attributes as keyword arguments.  Each node has  +    two attributes: `lineno` (the line number of the node) and `environment`.  +    The `environment` attribute is set at the end of the parsing process for  +    all nodes automatically.  +    """       fields: t.Tuple[str, ...] = ()      attributes: t.Tuple[str, ...] = ("lineno", "environment") -    abstract = True - +    abstract = True  +       lineno: int      environment: t.Optional["Environment"]      def __init__(self, *fields: t.Any, **attributes: t.Any) -> None: -        if self.abstract: +        if self.abstract:               raise TypeError("abstract nodes are not instantiable") -        if fields: -            if len(fields) != len(self.fields): -                if not self.fields: +        if fields:  +            if len(fields) != len(self.fields):  +                if not self.fields:                       raise TypeError(f"{type(self).__name__!r} takes 0 arguments")                  raise TypeError(                      f"{type(self).__name__!r} takes 0 or {len(self.fields)}"                      f" argument{'s' if len(self.fields) != 1 else ''}"                  )              for name, arg in zip(self.fields, fields): -                setattr(self, name, arg) -        for attr in self.attributes: -            setattr(self, attr, attributes.pop(attr, None)) -        if attributes: +                setattr(self, name, arg)  +        for attr in self.attributes:  +            setattr(self, attr, attributes.pop(attr, None))  +        if attributes:               raise TypeError(f"unknown attribute {next(iter(attributes))!r}") - +       def iter_fields(          self,          exclude: t.Optional[t.Container[str]] = None,          only: t.Optional[t.Container[str]] = None,      ) -> t.Iterator[t.Tuple[str, t.Any]]: -        """This method iterates over all fields that are defined and yields -        ``(key, value)`` tuples.  Per default all fields are returned, but -        it's possible to limit that to some fields by providing the `only` -        parameter or to exclude some using the `exclude` parameter.  Both -        should be sets or tuples of field names. -        """ -        for name in self.fields: +        """This method iterates over all fields that are defined and yields  +        ``(key, value)`` tuples.  Per default all fields are returned, but  +        it's possible to limit that to some fields by providing the `only`  +        parameter or to exclude some using the `exclude` parameter.  Both  +        should be sets or tuples of field names.  +        """  +        for name in self.fields:               if (                  (exclude is None and only is None)                  or (exclude is not None and name not in exclude)                  or (only is not None and name in only)              ): -                try: -                    yield name, getattr(self, name) -                except AttributeError: -                    pass - +                try:  +                    yield name, getattr(self, name)  +                except AttributeError:  +                    pass  +       def iter_child_nodes(          self,          exclude: t.Optional[t.Container[str]] = None,          only: t.Optional[t.Container[str]] = None,      ) -> t.Iterator["Node"]: -        """Iterates over all direct child nodes of the node.  This iterates -        over all fields and yields the values of they are nodes.  If the value -        of a field is a list all the nodes in that list are returned. -        """ +        """Iterates over all direct child nodes of the node.  This iterates  +        over all fields and yields the values of they are nodes.  If the value  +        of a field is a list all the nodes in that list are returned.  +        """           for _, item in self.iter_fields(exclude, only): -            if isinstance(item, list): -                for n in item: -                    if isinstance(n, Node): -                        yield n -            elif isinstance(item, Node): -                yield item - +            if isinstance(item, list):  +                for n in item:  +                    if isinstance(n, Node):  +                        yield n  +            elif isinstance(item, Node):  +                yield item  +       def find(self, node_type: t.Type[_NodeBound]) -> t.Optional[_NodeBound]: -        """Find the first node of a given type.  If no such node exists the -        return value is `None`. -        """ -        for result in self.find_all(node_type): -            return result - +        """Find the first node of a given type.  If no such node exists the  +        return value is `None`.  +        """  +        for result in self.find_all(node_type):  +            return result  +           return None      def find_all(          self, node_type: t.Union[t.Type[_NodeBound], t.Tuple[t.Type[_NodeBound], ...]]      ) -> t.Iterator[_NodeBound]: -        """Find all the nodes of a given type.  If the type is a tuple, -        the check is performed for any of the tuple items. -        """ -        for child in self.iter_child_nodes(): -            if isinstance(child, node_type): +        """Find all the nodes of a given type.  If the type is a tuple,  +        the check is performed for any of the tuple items.  +        """  +        for child in self.iter_child_nodes():  +            if isinstance(child, node_type):                   yield child  # type: ignore              yield from child.find_all(node_type) - +       def set_ctx(self, ctx: str) -> "Node": -        """Reset the context of a node and all child nodes.  Per default the -        parser will all generate nodes that have a 'load' context as it's the -        most common one.  This method is used in the parser to set assignment -        targets and other nodes to a store context. -        """ -        todo = deque([self]) -        while todo: -            node = todo.popleft() +        """Reset the context of a node and all child nodes.  Per default the  +        parser will all generate nodes that have a 'load' context as it's the  +        most common one.  This method is used in the parser to set assignment  +        targets and other nodes to a store context.  +        """  +        todo = deque([self])  +        while todo:  +            node = todo.popleft()               if "ctx" in node.fields:                  node.ctx = ctx  # type: ignore -            todo.extend(node.iter_child_nodes()) -        return self - +            todo.extend(node.iter_child_nodes())  +        return self  +       def set_lineno(self, lineno: int, override: bool = False) -> "Node": -        """Set the line numbers of the node and children.""" -        todo = deque([self]) -        while todo: -            node = todo.popleft() +        """Set the line numbers of the node and children."""  +        todo = deque([self])  +        while todo:  +            node = todo.popleft()               if "lineno" in node.attributes: -                if node.lineno is None or override: -                    node.lineno = lineno -            todo.extend(node.iter_child_nodes()) -        return self - +                if node.lineno is None or override:  +                    node.lineno = lineno  +            todo.extend(node.iter_child_nodes())  +        return self  +       def set_environment(self, environment: "Environment") -> "Node": -        """Set the environment for all nodes.""" -        todo = deque([self]) -        while todo: -            node = todo.popleft() -            node.environment = environment -            todo.extend(node.iter_child_nodes()) -        return self - +        """Set the environment for all nodes."""  +        todo = deque([self])  +        while todo:  +            node = todo.popleft()  +            node.environment = environment  +            todo.extend(node.iter_child_nodes())  +        return self  +       def __eq__(self, other: t.Any) -> bool:          if type(self) is not type(other):              return NotImplemented - +           return tuple(self.iter_fields()) == tuple(other.iter_fields()) - +       __hash__ = object.__hash__ - +       def __repr__(self) -> str:          args_str = ", ".join(f"{a}={getattr(self, a, None)!r}" for a in self.fields)          return f"{type(self).__name__}({args_str})" - +       def dump(self) -> str:          def _dump(node: t.Union[Node, t.Any]) -> None: -            if not isinstance(node, Node): -                buf.append(repr(node)) -                return - +            if not isinstance(node, Node):  +                buf.append(repr(node))  +                return  +               buf.append(f"nodes.{type(node).__name__}(") -            if not node.fields: +            if not node.fields:                   buf.append(")") -                return -            for idx, field in enumerate(node.fields): -                if idx: +                return  +            for idx, field in enumerate(node.fields):  +                if idx:                       buf.append(", ") -                value = getattr(node, field) -                if isinstance(value, list): +                value = getattr(node, field)  +                if isinstance(value, list):                       buf.append("[") -                    for idx, item in enumerate(value): -                        if idx: +                    for idx, item in enumerate(value):  +                        if idx:                               buf.append(", ") -                        _dump(item) +                        _dump(item)                       buf.append("]") -                else: -                    _dump(value) +                else:  +                    _dump(value)               buf.append(")")          buf: t.List[str] = [] -        _dump(self) +        _dump(self)           return "".join(buf) - - -class Stmt(Node): -    """Base node for all statements.""" - -    abstract = True - - -class Helper(Node): -    """Nodes that exist in a specific context only.""" - -    abstract = True - - -class Template(Node): -    """Node that represents a template.  This must be the outermost node that -    is passed to the compiler. -    """ - +  +  +class Stmt(Node):  +    """Base node for all statements."""  + +    abstract = True  +  +  +class Helper(Node):  +    """Nodes that exist in a specific context only."""  + +    abstract = True  +  +  +class Template(Node):  +    """Node that represents a template.  This must be the outermost node that  +    is passed to the compiler.  +    """  +       fields = ("body",)      body: t.List[Node] +  - -class Output(Stmt): -    """A node that holds multiple expressions which are then printed out. -    This is used both for the `print` statement and the regular template data. -    """ - +class Output(Stmt):  +    """A node that holds multiple expressions which are then printed out.  +    This is used both for the `print` statement and the regular template data.  +    """  +       fields = ("nodes",)      nodes: t.List["Expr"] +  - -class Extends(Stmt): -    """Represents an extends statement.""" - +class Extends(Stmt):  +    """Represents an extends statement."""  +       fields = ("template",)      template: "Expr" - - -class For(Stmt): -    """The for loop.  `target` is the target for the iteration (usually a -    :class:`Name` or :class:`Tuple`), `iter` the iterable.  `body` is a list -    of nodes that are used as loop-body, and `else_` a list of nodes for the -    `else` block.  If no else node exists it has to be an empty list. - -    For filtered nodes an expression can be stored as `test`, otherwise `None`. -    """ - +  + +class For(Stmt):  +    """The for loop.  `target` is the target for the iteration (usually a  +    :class:`Name` or :class:`Tuple`), `iter` the iterable.  `body` is a list  +    of nodes that are used as loop-body, and `else_` a list of nodes for the  +    `else` block.  If no else node exists it has to be an empty list.  +  +    For filtered nodes an expression can be stored as `test`, otherwise `None`.  +    """  +       fields = ("target", "iter", "body", "else_", "test", "recursive")      target: Node      iter: Node @@ -330,71 +330,71 @@ class For(Stmt):      else_: t.List[Node]      test: t.Optional[Node]      recursive: bool +  - -class If(Stmt): -    """If `test` is true, `body` is rendered, else `else_`.""" - +class If(Stmt):  +    """If `test` is true, `body` is rendered, else `else_`."""  +       fields = ("test", "body", "elif_", "else_")      test: Node      body: t.List[Node]      elif_: t.List["If"]      else_: t.List[Node] - - -class Macro(Stmt): -    """A macro definition.  `name` is the name of the macro, `args` a list of -    arguments and `defaults` a list of defaults if there are any.  `body` is -    a list of nodes for the macro body. -    """ - +  + +class Macro(Stmt):  +    """A macro definition.  `name` is the name of the macro, `args` a list of  +    arguments and `defaults` a list of defaults if there are any.  `body` is  +    a list of nodes for the macro body.  +    """  +       fields = ("name", "args", "defaults", "body")      name: str      args: t.List["Name"]      defaults: t.List["Expr"]      body: t.List[Node] +  - -class CallBlock(Stmt): -    """Like a macro without a name but a call instead.  `call` is called with -    the unnamed macro as `caller` argument this node holds. -    """ - +class CallBlock(Stmt):  +    """Like a macro without a name but a call instead.  `call` is called with  +    the unnamed macro as `caller` argument this node holds.  +    """  +       fields = ("call", "args", "defaults", "body")      call: "Call"      args: t.List["Name"]      defaults: t.List["Expr"]      body: t.List[Node] +  - -class FilterBlock(Stmt): -    """Node for filter sections.""" - +class FilterBlock(Stmt):  +    """Node for filter sections."""  +       fields = ("body", "filter")      body: t.List[Node]      filter: "Filter" - - -class With(Stmt): -    """Specific node for with statements.  In older versions of Jinja the -    with statement was implemented on the base of the `Scope` node instead. - -    .. versionadded:: 2.9.3 -    """ - +  + +class With(Stmt):  +    """Specific node for with statements.  In older versions of Jinja the  +    with statement was implemented on the base of the `Scope` node instead.  +  +    .. versionadded:: 2.9.3  +    """  +       fields = ("targets", "values", "body")      targets: t.List["Expr"]      values: t.List["Expr"]      body: t.List[Node] +  - -class Block(Stmt): +class Block(Stmt):       """A node that represents a block. - +       .. versionchanged:: 3.0.0          the `required` field was added.      """ - +       fields = ("name", "body", "scoped", "required")      name: str      body: t.List[Node] @@ -402,272 +402,272 @@ class Block(Stmt):      required: bool -class Include(Stmt): -    """A node that represents the include tag.""" - +class Include(Stmt):  +    """A node that represents the include tag."""  +       fields = ("template", "with_context", "ignore_missing")      template: "Expr"      with_context: bool      ignore_missing: bool +  - -class Import(Stmt): -    """A node that represents the import tag.""" - +class Import(Stmt):  +    """A node that represents the import tag."""  +       fields = ("template", "target", "with_context")      template: "Expr"      target: str      with_context: bool - - -class FromImport(Stmt): -    """A node that represents the from import tag.  It's important to not -    pass unsafe names to the name attribute.  The compiler translates the -    attribute lookups directly into getattr calls and does *not* use the -    subscript callback of the interface.  As exported variables may not -    start with double underscores (which the parser asserts) this is not a -    problem for regular Jinja code, but if this node is used in an extension -    extra care must be taken. - -    The list of names may contain tuples if aliases are wanted. -    """ - +  + +class FromImport(Stmt):  +    """A node that represents the from import tag.  It's important to not  +    pass unsafe names to the name attribute.  The compiler translates the  +    attribute lookups directly into getattr calls and does *not* use the  +    subscript callback of the interface.  As exported variables may not  +    start with double underscores (which the parser asserts) this is not a  +    problem for regular Jinja code, but if this node is used in an extension  +    extra care must be taken.  +  +    The list of names may contain tuples if aliases are wanted.  +    """  +       fields = ("template", "names", "with_context")      template: "Expr"      names: t.List[t.Union[str, t.Tuple[str, str]]]      with_context: bool +  - -class ExprStmt(Stmt): -    """A statement that evaluates an expression and discards the result.""" - +class ExprStmt(Stmt):  +    """A statement that evaluates an expression and discards the result."""  +       fields = ("node",)      node: Node +  - -class Assign(Stmt): -    """Assigns an expression to a target.""" - +class Assign(Stmt):  +    """Assigns an expression to a target."""  +       fields = ("target", "node")      target: "Expr"      node: Node +  - -class AssignBlock(Stmt): -    """Assigns a block to a target.""" - +class AssignBlock(Stmt):  +    """Assigns a block to a target."""  +       fields = ("target", "filter", "body")      target: "Expr"      filter: t.Optional["Filter"]      body: t.List[Node] +  +class Expr(Node):  +    """Baseclass for all expressions."""  -class Expr(Node): -    """Baseclass for all expressions.""" - -    abstract = True - +    abstract = True  +       def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        """Return the value of the expression as constant or raise -        :exc:`Impossible` if this was not possible. - -        An :class:`EvalContext` can be provided, if none is given -        a default context is created which requires the nodes to have -        an attached environment. - -        .. versionchanged:: 2.4 -           the `eval_ctx` parameter was added. -        """ -        raise Impossible() - +        """Return the value of the expression as constant or raise  +        :exc:`Impossible` if this was not possible.  +  +        An :class:`EvalContext` can be provided, if none is given  +        a default context is created which requires the nodes to have  +        an attached environment.  +  +        .. versionchanged:: 2.4  +           the `eval_ctx` parameter was added.  +        """  +        raise Impossible()  +       def can_assign(self) -> bool: -        """Check if it's possible to assign something to this node.""" -        return False - - -class BinExpr(Expr): -    """Baseclass for all binary expressions.""" +        """Check if it's possible to assign something to this node."""  +        return False  +  +  +class BinExpr(Expr):  +    """Baseclass for all binary expressions."""       fields = ("left", "right")      left: Expr      right: Expr      operator: str -    abstract = True - +    abstract = True  +       def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) +        eval_ctx = get_eval_context(self, eval_ctx)  -        # intercepted operators cannot be folded at compile time +        # intercepted operators cannot be folded at compile time           if (              eval_ctx.environment.sandboxed              and self.operator in eval_ctx.environment.intercepted_binops  # type: ignore          ): -            raise Impossible() -        f = _binop_to_func[self.operator] -        try: -            return f(self.left.as_const(eval_ctx), self.right.as_const(eval_ctx)) +            raise Impossible()  +        f = _binop_to_func[self.operator]  +        try:  +            return f(self.left.as_const(eval_ctx), self.right.as_const(eval_ctx))           except Exception as e:              raise Impossible() from e - - -class UnaryExpr(Expr): -    """Baseclass for all unary expressions.""" +  +  +class UnaryExpr(Expr):  +    """Baseclass for all unary expressions."""       fields = ("node",)      node: Expr      operator: str -    abstract = True - +    abstract = True  +       def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) +        eval_ctx = get_eval_context(self, eval_ctx)  -        # intercepted operators cannot be folded at compile time +        # intercepted operators cannot be folded at compile time           if (              eval_ctx.environment.sandboxed              and self.operator in eval_ctx.environment.intercepted_unops  # type: ignore          ): -            raise Impossible() -        f = _uaop_to_func[self.operator] -        try: -            return f(self.node.as_const(eval_ctx)) +            raise Impossible()  +        f = _uaop_to_func[self.operator]  +        try:  +            return f(self.node.as_const(eval_ctx))           except Exception as e:              raise Impossible() from e - - -class Name(Expr): -    """Looks up a name or stores a value in a name. -    The `ctx` of the node can be one of the following values: - -    -   `store`: store a value in the name -    -   `load`: load that name -    -   `param`: like `store` but if the name was defined as function parameter. -    """ - +  +  +class Name(Expr):  +    """Looks up a name or stores a value in a name.  +    The `ctx` of the node can be one of the following values:  +  +    -   `store`: store a value in the name  +    -   `load`: load that name  +    -   `param`: like `store` but if the name was defined as function parameter.  +    """  +       fields = ("name", "ctx")      name: str      ctx: str      def can_assign(self) -> bool:          return self.name not in {"true", "false", "none", "True", "False", "None"} - - -class NSRef(Expr): -    """Reference to a namespace value assignment""" - +  +  +class NSRef(Expr):  +    """Reference to a namespace value assignment"""  +       fields = ("name", "attr")      name: str      attr: str      def can_assign(self) -> bool: -        # We don't need any special checks here; NSRef assignments have a -        # runtime check to ensure the target is a namespace object which will -        # have been checked already as it is created using a normal assignment -        # which goes through a `Name` node. -        return True - - -class Literal(Expr): -    """Baseclass for literals.""" - -    abstract = True - - -class Const(Literal): -    """All constant values.  The parser will return this node for simple -    constants such as ``42`` or ``"foo"`` but it can be used to store more -    complex values such as lists too.  Only constants with a safe -    representation (objects where ``eval(repr(x)) == x`` is true). -    """ - +        # We don't need any special checks here; NSRef assignments have a  +        # runtime check to ensure the target is a namespace object which will  +        # have been checked already as it is created using a normal assignment  +        # which goes through a `Name` node.  +        return True  +  +  +class Literal(Expr):  +    """Baseclass for literals."""  + +    abstract = True  +  +  +class Const(Literal):  +    """All constant values.  The parser will return this node for simple  +    constants such as ``42`` or ``"foo"`` but it can be used to store more  +    complex values such as lists too.  Only constants with a safe  +    representation (objects where ``eval(repr(x)) == x`` is true).  +    """  +       fields = ("value",)      value: t.Any      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any:          return self.value - -    @classmethod +  +    @classmethod       def from_untrusted(          cls,          value: t.Any,          lineno: t.Optional[int] = None,          environment: "t.Optional[Environment]" = None,      ) -> "Const": -        """Return a const object if the value is representable as -        constant value in the generated code, otherwise it will raise -        an `Impossible` exception. -        """ -        from .compiler import has_safe_repr - -        if not has_safe_repr(value): -            raise Impossible() -        return cls(value, lineno=lineno, environment=environment) - - -class TemplateData(Literal): -    """A constant template string.""" - +        """Return a const object if the value is representable as  +        constant value in the generated code, otherwise it will raise  +        an `Impossible` exception.  +        """  +        from .compiler import has_safe_repr  + +        if not has_safe_repr(value):  +            raise Impossible()  +        return cls(value, lineno=lineno, environment=environment)  +  +  +class TemplateData(Literal):  +    """A constant template string."""  +       fields = ("data",)      data: str      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> str: -        eval_ctx = get_eval_context(self, eval_ctx) -        if eval_ctx.volatile: -            raise Impossible() -        if eval_ctx.autoescape: -            return Markup(self.data) -        return self.data - - -class Tuple(Literal): -    """For loop unpacking and some other things like multiple arguments -    for subscripts.  Like for :class:`Name` `ctx` specifies if the tuple -    is used for loading the names or storing. -    """ - +        eval_ctx = get_eval_context(self, eval_ctx)  +        if eval_ctx.volatile:  +            raise Impossible()  +        if eval_ctx.autoescape:  +            return Markup(self.data)  +        return self.data  +  +  +class Tuple(Literal):  +    """For loop unpacking and some other things like multiple arguments  +    for subscripts.  Like for :class:`Name` `ctx` specifies if the tuple  +    is used for loading the names or storing.  +    """  +       fields = ("items", "ctx")      items: t.List[Expr]      ctx: str      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Tuple[t.Any, ...]: -        eval_ctx = get_eval_context(self, eval_ctx) -        return tuple(x.as_const(eval_ctx) for x in self.items) - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return tuple(x.as_const(eval_ctx) for x in self.items)  +       def can_assign(self) -> bool: -        for item in self.items: -            if not item.can_assign(): -                return False -        return True - - -class List(Literal): -    """Any list literal such as ``[1, 2, 3]``""" - +        for item in self.items:  +            if not item.can_assign():  +                return False  +        return True  +  +  +class List(Literal):  +    """Any list literal such as ``[1, 2, 3]``"""  +       fields = ("items",)      items: t.List[Expr]      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.List[t.Any]: -        eval_ctx = get_eval_context(self, eval_ctx) -        return [x.as_const(eval_ctx) for x in self.items] - - -class Dict(Literal): -    """Any dict literal such as ``{1: 2, 3: 4}``.  The items must be a list of -    :class:`Pair` nodes. -    """ - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return [x.as_const(eval_ctx) for x in self.items]  +  +  +class Dict(Literal):  +    """Any dict literal such as ``{1: 2, 3: 4}``.  The items must be a list of  +    :class:`Pair` nodes.  +    """  +       fields = ("items",)      items: t.List["Pair"]      def as_const(          self, eval_ctx: t.Optional[EvalContext] = None      ) -> t.Dict[t.Any, t.Any]: -        eval_ctx = get_eval_context(self, eval_ctx) -        return dict(x.as_const(eval_ctx) for x in self.items) - - -class Pair(Helper): -    """A key, value pair for dicts.""" - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return dict(x.as_const(eval_ctx) for x in self.items)  +  +  +class Pair(Helper):  +    """A key, value pair for dicts."""  +       fields = ("key", "value")      key: Expr      value: Expr @@ -675,65 +675,65 @@ class Pair(Helper):      def as_const(          self, eval_ctx: t.Optional[EvalContext] = None      ) -> t.Tuple[t.Any, t.Any]: -        eval_ctx = get_eval_context(self, eval_ctx) -        return self.key.as_const(eval_ctx), self.value.as_const(eval_ctx) - - -class Keyword(Helper): -    """A key, value pair for keyword arguments where key is a string.""" - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return self.key.as_const(eval_ctx), self.value.as_const(eval_ctx)  +  +  +class Keyword(Helper):  +    """A key, value pair for keyword arguments where key is a string."""  +       fields = ("key", "value")      key: str      value: Expr      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Tuple[str, t.Any]: -        eval_ctx = get_eval_context(self, eval_ctx) -        return self.key, self.value.as_const(eval_ctx) - - -class CondExpr(Expr): -    """A conditional expression (inline if expression).  (``{{ -    foo if bar else baz }}``) -    """ - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return self.key, self.value.as_const(eval_ctx)  +  +  +class CondExpr(Expr):  +    """A conditional expression (inline if expression).  (``{{  +    foo if bar else baz }}``)  +    """  +       fields = ("test", "expr1", "expr2")      test: Expr      expr1: Expr      expr2: t.Optional[Expr]      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) -        if self.test.as_const(eval_ctx): -            return self.expr1.as_const(eval_ctx) - -        # if we evaluate to an undefined object, we better do that at runtime -        if self.expr2 is None: -            raise Impossible() - -        return self.expr2.as_const(eval_ctx) - - +        eval_ctx = get_eval_context(self, eval_ctx)  +        if self.test.as_const(eval_ctx):  +            return self.expr1.as_const(eval_ctx)  +  +        # if we evaluate to an undefined object, we better do that at runtime  +        if self.expr2 is None:  +            raise Impossible()  +  +        return self.expr2.as_const(eval_ctx)  +  +   def args_as_const(      node: t.Union["_FilterTestCommon", "Call"], eval_ctx: t.Optional[EvalContext]  ) -> t.Tuple[t.List[t.Any], t.Dict[t.Any, t.Any]]: -    args = [x.as_const(eval_ctx) for x in node.args] -    kwargs = dict(x.as_const(eval_ctx) for x in node.kwargs) - -    if node.dyn_args is not None: -        try: -            args.extend(node.dyn_args.as_const(eval_ctx)) +    args = [x.as_const(eval_ctx) for x in node.args]  +    kwargs = dict(x.as_const(eval_ctx) for x in node.kwargs)  +  +    if node.dyn_args is not None:  +        try:  +            args.extend(node.dyn_args.as_const(eval_ctx))           except Exception as e:              raise Impossible() from e - -    if node.dyn_kwargs is not None: -        try: -            kwargs.update(node.dyn_kwargs.as_const(eval_ctx)) +  +    if node.dyn_kwargs is not None:  +        try:  +            kwargs.update(node.dyn_kwargs.as_const(eval_ctx))           except Exception as e:              raise Impossible() from e - -    return args, kwargs - - +  +    return args, kwargs  +  +   class _FilterTestCommon(Expr):      fields = ("node", "name", "args", "kwargs", "dyn_args", "dyn_kwargs")      node: Expr @@ -744,65 +744,65 @@ class _FilterTestCommon(Expr):      dyn_kwargs: t.Optional[Expr]      abstract = True      _is_filter = True - +       def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) - +        eval_ctx = get_eval_context(self, eval_ctx)  +           if eval_ctx.volatile: -            raise Impossible() - +            raise Impossible()  +           if self._is_filter:              env_map = eval_ctx.environment.filters          else:              env_map = eval_ctx.environment.tests - +           func = env_map.get(self.name)          pass_arg = _PassArg.from_obj(func)  # type: ignore          if func is None or pass_arg is _PassArg.context: -            raise Impossible() - +            raise Impossible()  +           if eval_ctx.environment.is_async and (              getattr(func, "jinja_async_variant", False) is True              or inspect.iscoroutinefunction(func) -        ): -            raise Impossible() - -        args, kwargs = args_as_const(self, eval_ctx) -        args.insert(0, self.node.as_const(eval_ctx)) - +        ):  +            raise Impossible()  +  +        args, kwargs = args_as_const(self, eval_ctx)  +        args.insert(0, self.node.as_const(eval_ctx))  +           if pass_arg is _PassArg.eval_context: -            args.insert(0, eval_ctx) +            args.insert(0, eval_ctx)           elif pass_arg is _PassArg.environment:              args.insert(0, eval_ctx.environment) - -        try: +  +        try:               return func(*args, **kwargs)          except Exception as e:              raise Impossible() from e - - +  +   class Filter(_FilterTestCommon):      """Apply a filter to an expression. ``name`` is the name of the      filter, the other fields are the same as :class:`Call`.      If ``node`` is ``None``, the filter is being used in a filter block      and is applied to the content of the block. -    """ - +    """  +       node: t.Optional[Expr]  # type: ignore - +       def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any:          if self.node is None: -            raise Impossible() - +            raise Impossible()  +           return super().as_const(eval_ctx=eval_ctx) - - +  +   class Test(_FilterTestCommon):      """Apply a test to an expression. ``name`` is the name of the test,      the other field are the same as :class:`Call`. - +       .. versionchanged:: 3.0          ``as_const`` shares the same logic for filters and tests. Tests          check for volatile, async, and ``@pass_context`` etc. @@ -812,25 +812,25 @@ class Test(_FilterTestCommon):      _is_filter = False -class Call(Expr): -    """Calls an expression.  `args` is a list of arguments, `kwargs` a list -    of keyword arguments (list of :class:`Keyword` nodes), and `dyn_args` -    and `dyn_kwargs` has to be either `None` or a node that is used as -    node for dynamic positional (``*args``) or keyword (``**kwargs``) -    arguments. -    """ - +class Call(Expr):  +    """Calls an expression.  `args` is a list of arguments, `kwargs` a list  +    of keyword arguments (list of :class:`Keyword` nodes), and `dyn_args`  +    and `dyn_kwargs` has to be either `None` or a node that is used as  +    node for dynamic positional (``*args``) or keyword (``**kwargs``)  +    arguments.  +    """  +       fields = ("node", "args", "kwargs", "dyn_args", "dyn_kwargs")      node: Expr      args: t.List[Expr]      kwargs: t.List[Keyword]      dyn_args: t.Optional[Expr]      dyn_kwargs: t.Optional[Expr] +  - -class Getitem(Expr): -    """Get an attribute or item from an expression and prefer the item.""" - +class Getitem(Expr):  +    """Get an attribute or item from an expression and prefer the item."""  +       fields = ("node", "arg", "ctx")      node: Expr      arg: Expr @@ -838,23 +838,23 @@ class Getitem(Expr):      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any:          if self.ctx != "load": -            raise Impossible() +            raise Impossible()           eval_ctx = get_eval_context(self, eval_ctx) -        try: +        try:               return eval_ctx.environment.getitem(                  self.node.as_const(eval_ctx), self.arg.as_const(eval_ctx)              )          except Exception as e:              raise Impossible() from e - - -class Getattr(Expr): -    """Get an attribute or item from an expression that is a ascii-only -    bytestring and prefer the attribute. -    """ - +  +  +class Getattr(Expr):  +    """Get an attribute or item from an expression that is a ascii-only  +    bytestring and prefer the attribute.  +    """  +       fields = ("node", "attr", "ctx")      node: Expr      attr: str @@ -862,212 +862,212 @@ class Getattr(Expr):      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any:          if self.ctx != "load": -            raise Impossible() +            raise Impossible()           eval_ctx = get_eval_context(self, eval_ctx) -        try: +        try:               return eval_ctx.environment.getattr(self.node.as_const(eval_ctx), self.attr)          except Exception as e:              raise Impossible() from e - - -class Slice(Expr): -    """Represents a slice object.  This must only be used as argument for -    :class:`Subscript`. -    """ - +  +  +class Slice(Expr):  +    """Represents a slice object.  This must only be used as argument for  +    :class:`Subscript`.  +    """  +       fields = ("start", "stop", "step")      start: t.Optional[Expr]      stop: t.Optional[Expr]      step: t.Optional[Expr]      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> slice: -        eval_ctx = get_eval_context(self, eval_ctx) +        eval_ctx = get_eval_context(self, eval_ctx)           def const(obj: t.Optional[Expr]) -> t.Optional[t.Any]: -            if obj is None: -                return None -            return obj.as_const(eval_ctx) - -        return slice(const(self.start), const(self.stop), const(self.step)) - - -class Concat(Expr): +            if obj is None:  +                return None  +            return obj.as_const(eval_ctx)  + +        return slice(const(self.start), const(self.stop), const(self.step))  +  +  +class Concat(Expr):       """Concatenates the list of expressions provided after converting      them to strings. -    """ - +    """  +       fields = ("nodes",)      nodes: t.List[Expr]      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> str: -        eval_ctx = get_eval_context(self, eval_ctx) +        eval_ctx = get_eval_context(self, eval_ctx)           return "".join(str(x.as_const(eval_ctx)) for x in self.nodes) - - -class Compare(Expr): -    """Compares an expression with some other expressions.  `ops` must be a -    list of :class:`Operand`\\s. -    """ - +  +  +class Compare(Expr):  +    """Compares an expression with some other expressions.  `ops` must be a  +    list of :class:`Operand`\\s.  +    """  +       fields = ("expr", "ops")      expr: Expr      ops: t.List["Operand"]      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) -        result = value = self.expr.as_const(eval_ctx) +        eval_ctx = get_eval_context(self, eval_ctx)  +        result = value = self.expr.as_const(eval_ctx)  -        try: -            for op in self.ops: -                new_value = op.expr.as_const(eval_ctx) -                result = _cmpop_to_func[op.op](value, new_value) +        try:  +            for op in self.ops:  +                new_value = op.expr.as_const(eval_ctx)  +                result = _cmpop_to_func[op.op](value, new_value)                   if not result:                      return False -                value = new_value +                value = new_value           except Exception as e:              raise Impossible() from e -        return result - - -class Operand(Helper): -    """Holds an operator and an expression.""" - +        return result  +  +  +class Operand(Helper):  +    """Holds an operator and an expression."""  +       fields = ("op", "expr")      op: str      expr: Expr -class Mul(BinExpr): -    """Multiplies the left with the right node.""" - +class Mul(BinExpr):  +    """Multiplies the left with the right node."""  +       operator = "*" +  - -class Div(BinExpr): -    """Divides the left by the right node.""" - +class Div(BinExpr):  +    """Divides the left by the right node."""  +       operator = "/" +  - -class FloorDiv(BinExpr): +class FloorDiv(BinExpr):       """Divides the left by the right node and converts the -    result into an integer by truncating. -    """ - +    result into an integer by truncating.  +    """  +       operator = "//" +  - -class Add(BinExpr): -    """Add the left to the right node.""" - +class Add(BinExpr):  +    """Add the left to the right node."""  +       operator = "+" +  - -class Sub(BinExpr): -    """Subtract the right from the left node.""" - +class Sub(BinExpr):  +    """Subtract the right from the left node."""  +       operator = "-" +  - -class Mod(BinExpr): -    """Left modulo right.""" - +class Mod(BinExpr):  +    """Left modulo right."""  +       operator = "%" +  - -class Pow(BinExpr): -    """Left to the power of right.""" - +class Pow(BinExpr):  +    """Left to the power of right."""  +       operator = "**" +  - -class And(BinExpr): -    """Short circuited AND.""" - +class And(BinExpr):  +    """Short circuited AND."""  +       operator = "and"      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) -        return self.left.as_const(eval_ctx) and self.right.as_const(eval_ctx) - - -class Or(BinExpr): -    """Short circuited OR.""" - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return self.left.as_const(eval_ctx) and self.right.as_const(eval_ctx)  +  +  +class Or(BinExpr):  +    """Short circuited OR."""  +       operator = "or"      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> t.Any: -        eval_ctx = get_eval_context(self, eval_ctx) -        return self.left.as_const(eval_ctx) or self.right.as_const(eval_ctx) - - -class Not(UnaryExpr): -    """Negate the expression.""" - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return self.left.as_const(eval_ctx) or self.right.as_const(eval_ctx)  +  +  +class Not(UnaryExpr):  +    """Negate the expression."""  +       operator = "not" +  - -class Neg(UnaryExpr): -    """Make the expression negative.""" - +class Neg(UnaryExpr):  +    """Make the expression negative."""  +       operator = "-" +  - -class Pos(UnaryExpr): -    """Make the expression positive (noop for most expressions)""" - +class Pos(UnaryExpr):  +    """Make the expression positive (noop for most expressions)"""  +       operator = "+" - - -# Helpers for extensions - - -class EnvironmentAttribute(Expr): -    """Loads an attribute from the environment object.  This is useful for -    extensions that want to call a callback stored on the environment. -    """ - +  + +# Helpers for extensions  +  +  +class EnvironmentAttribute(Expr):  +    """Loads an attribute from the environment object.  This is useful for  +    extensions that want to call a callback stored on the environment.  +    """  +       fields = ("name",)      name: str - - -class ExtensionAttribute(Expr): -    """Returns the attribute of an extension bound to the environment. -    The identifier is the identifier of the :class:`Extension`. - -    This node is usually constructed by calling the -    :meth:`~jinja2.ext.Extension.attr` method on an extension. -    """ - +  + +class ExtensionAttribute(Expr):  +    """Returns the attribute of an extension bound to the environment.  +    The identifier is the identifier of the :class:`Extension`.  +  +    This node is usually constructed by calling the  +    :meth:`~jinja2.ext.Extension.attr` method on an extension.  +    """  +       fields = ("identifier", "name")      identifier: str      name: str - - -class ImportedName(Expr): -    """If created with an import name the import name is returned on node -    access.  For example ``ImportedName('cgi.escape')`` returns the `escape` -    function from the cgi module on evaluation.  Imports are optimized by the -    compiler so there is no need to assign them to local variables. -    """ - +  + +class ImportedName(Expr):  +    """If created with an import name the import name is returned on node  +    access.  For example ``ImportedName('cgi.escape')`` returns the `escape`  +    function from the cgi module on evaluation.  Imports are optimized by the  +    compiler so there is no need to assign them to local variables.  +    """  +       fields = ("importname",)      importname: str +  - -class InternalName(Expr): -    """An internal name in the compiler.  You cannot create these nodes -    yourself but the parser provides a -    :meth:`~jinja2.parser.Parser.free_identifier` method that creates -    a new identifier for you.  This identifier is not available from the +class InternalName(Expr):  +    """An internal name in the compiler.  You cannot create these nodes  +    yourself but the parser provides a  +    :meth:`~jinja2.parser.Parser.free_identifier` method that creates  +    a new identifier for you.  This identifier is not available from the       template and is not treated specially by the compiler. -    """ - +    """  +       fields = ("name",)      name: str @@ -1076,59 +1076,59 @@ class InternalName(Expr):              "Can't create internal names.  Use the "              "`free_identifier` method on a parser."          ) - - -class MarkSafe(Expr): -    """Mark the wrapped expression as safe (wrap it as `Markup`).""" - +  +  +class MarkSafe(Expr):  +    """Mark the wrapped expression as safe (wrap it as `Markup`)."""  +       fields = ("expr",)      expr: Expr      def as_const(self, eval_ctx: t.Optional[EvalContext] = None) -> Markup: -        eval_ctx = get_eval_context(self, eval_ctx) -        return Markup(self.expr.as_const(eval_ctx)) - - -class MarkSafeIfAutoescape(Expr): -    """Mark the wrapped expression as safe (wrap it as `Markup`) but -    only if autoescaping is active. - -    .. versionadded:: 2.5 -    """ - +        eval_ctx = get_eval_context(self, eval_ctx)  +        return Markup(self.expr.as_const(eval_ctx))  +  +  +class MarkSafeIfAutoescape(Expr):  +    """Mark the wrapped expression as safe (wrap it as `Markup`) but  +    only if autoescaping is active.  +  +    .. versionadded:: 2.5  +    """  +       fields = ("expr",)      expr: Expr      def as_const(          self, eval_ctx: t.Optional[EvalContext] = None      ) -> t.Union[Markup, t.Any]: -        eval_ctx = get_eval_context(self, eval_ctx) -        if eval_ctx.volatile: -            raise Impossible() -        expr = self.expr.as_const(eval_ctx) -        if eval_ctx.autoescape: -            return Markup(expr) -        return expr - - -class ContextReference(Expr): -    """Returns the current template context.  It can be used like a -    :class:`Name` node, with a ``'load'`` ctx and will return the -    current :class:`~jinja2.runtime.Context` object. - -    Here an example that assigns the current template name to a -    variable named `foo`:: - -        Assign(Name('foo', ctx='store'), -               Getattr(ContextReference(), 'name')) +        eval_ctx = get_eval_context(self, eval_ctx)  +        if eval_ctx.volatile:  +            raise Impossible()  +        expr = self.expr.as_const(eval_ctx)  +        if eval_ctx.autoescape:  +            return Markup(expr)  +        return expr  +  +  +class ContextReference(Expr):  +    """Returns the current template context.  It can be used like a  +    :class:`Name` node, with a ``'load'`` ctx and will return the  +    current :class:`~jinja2.runtime.Context` object.  +  +    Here an example that assigns the current template name to a  +    variable named `foo`::  +  +        Assign(Name('foo', ctx='store'),  +               Getattr(ContextReference(), 'name'))       This is basically equivalent to using the      :func:`~jinja2.pass_context` decorator when using the high-level      API, which causes a reference to the context to be passed as the      first argument to a function. -    """ - - +    """  +  +   class DerivedContextReference(Expr):      """Return the current template context including locals. Behaves      exactly like :class:`ContextReference`, but includes local @@ -1138,64 +1138,64 @@ class DerivedContextReference(Expr):      """ -class Continue(Stmt): -    """Continue a loop.""" - - -class Break(Stmt): -    """Break a loop.""" - - -class Scope(Stmt): -    """An artificial scope.""" - +class Continue(Stmt):  +    """Continue a loop."""  +  +  +class Break(Stmt):  +    """Break a loop."""  +  +  +class Scope(Stmt):  +    """An artificial scope."""  +       fields = ("body",)      body: t.List[Node] - - -class OverlayScope(Stmt): -    """An overlay scope for extensions.  This is a largely unoptimized scope -    that however can be used to introduce completely arbitrary variables into -    a sub scope from a dictionary or dictionary like object.  The `context` -    field has to evaluate to a dictionary object. - -    Example usage:: - -        OverlayScope(context=self.call_method('get_context'), -                     body=[...]) - -    .. versionadded:: 2.10 -    """ - +  + +class OverlayScope(Stmt):  +    """An overlay scope for extensions.  This is a largely unoptimized scope  +    that however can be used to introduce completely arbitrary variables into  +    a sub scope from a dictionary or dictionary like object.  The `context`  +    field has to evaluate to a dictionary object.  +  +    Example usage::  +  +        OverlayScope(context=self.call_method('get_context'),  +                     body=[...])  +  +    .. versionadded:: 2.10  +    """  +       fields = ("context", "body")      context: Expr      body: t.List[Node] - - -class EvalContextModifier(Stmt): -    """Modifies the eval context.  For each option that should be modified, -    a :class:`Keyword` has to be added to the :attr:`options` list. - -    Example to change the `autoescape` setting:: - -        EvalContextModifier(options=[Keyword('autoescape', Const(True))]) -    """ - +  + +class EvalContextModifier(Stmt):  +    """Modifies the eval context.  For each option that should be modified,  +    a :class:`Keyword` has to be added to the :attr:`options` list.  +  +    Example to change the `autoescape` setting::  +  +        EvalContextModifier(options=[Keyword('autoescape', Const(True))])  +    """  +       fields = ("options",)      options: t.List[Keyword] - - -class ScopedEvalContextModifier(EvalContextModifier): -    """Modifies the eval context and reverts it later.  Works exactly like -    :class:`EvalContextModifier` but will only modify the -    :class:`~jinja2.nodes.EvalContext` for nodes in the :attr:`body`. -    """ - +  + +class ScopedEvalContextModifier(EvalContextModifier):  +    """Modifies the eval context and reverts it later.  Works exactly like  +    :class:`EvalContextModifier` but will only modify the  +    :class:`~jinja2.nodes.EvalContext` for nodes in the :attr:`body`.  +    """  +       fields = ("body",)      body: t.List[Node] +  - -# make sure nobody creates custom nodes +# make sure nobody creates custom nodes   def _failing_new(*args: t.Any, **kwargs: t.Any) -> "te.NoReturn":      raise TypeError("can't create custom node types") | 
