From 50e0f5e79178f3eeb5b4d84b088f377d8ab94627 Mon Sep 17 00:00:00 2001 From: Georg Osang Date: Wed, 8 Feb 2023 10:29:05 +0100 Subject: [PATCH] FlowParser: Don't parse templates for excluded rows --- parsers/common/cellparser.py | 14 +++++++++----- parsers/common/sheetparser.py | 5 +++-- parsers/common/tests/mock_sheetparser.py | 2 +- parsers/creation/flowparser.py | 4 ++-- 4 files changed, 15 insertions(+), 10 deletions(-) diff --git a/parsers/common/cellparser.py b/parsers/common/cellparser.py index 1a87e5c..5d960e4 100644 --- a/parsers/common/cellparser.py +++ b/parsers/common/cellparser.py @@ -67,16 +67,18 @@ def parse(self, value, context={}): return self.split_into_lists(value) def parse_as_string(self, value, context={}, is_object=None): + # If context is None, template parsing is omitted entirely. # is_object is a pass-by-reference boolean, realised via # the class BooleanWrapper, to indicate to the caller # whether the parsing result represents an object that # is not to be processed any further. if value is None: return '' - if not context and '{' not in value: + if context is None or (not context and '{' not in value): # This is a hacky optimization. return value stripped_value = value.strip() + env = self.env if stripped_value.startswith('{@') and stripped_value.endswith('@}'): # Special case: Return a python object rather than a string, # if possible. @@ -84,8 +86,10 @@ def parse_as_string(self, value, context={}, is_object=None): assert stripped_value[2:].find('{@') == -1 if is_object is not None: is_object.boolean = True - template = self.native_env.from_string(stripped_value) - return template.render(context) - else: - template = self.env.from_string(stripped_value) + env = self.native_env + + try: + template = env.from_string(stripped_value) return template.render(context) + except Exception as e: + raise ValueError(str(e), stripped_value, context) diff --git a/parsers/common/sheetparser.py b/parsers/common/sheetparser.py index 0e20b7e..9317146 100644 --- a/parsers/common/sheetparser.py +++ b/parsers/common/sheetparser.py @@ -35,12 +35,13 @@ def go_to_bookmark(self, name): def remove_bookmark(self, name): self.bookmarks.pop(name) - def parse_next_row(self): + def parse_next_row(self, omit_templating=False): try: input_row = next(self.iterator) except StopIteration: return None - row = self.row_parser.parse_row(input_row, self.context) + context = self.context if not omit_templating else None + row = self.row_parser.parse_row(input_row, context) return row def parse_all(self): diff --git a/parsers/common/tests/mock_sheetparser.py b/parsers/common/tests/mock_sheetparser.py index b35d979..af7686e 100644 --- a/parsers/common/tests/mock_sheetparser.py +++ b/parsers/common/tests/mock_sheetparser.py @@ -17,7 +17,7 @@ def __init__(self, row_parser, rows, context={}): self.iterator = iter(self.input_rows) self.context = copy.deepcopy(context) - def parse_next_row(self): + def parse_next_row(self, omit_templating=False): # Simply return the input. try: input_row = next(self.iterator) diff --git a/parsers/creation/flowparser.py b/parsers/creation/flowparser.py index 3e5a41e..588154d 100644 --- a/parsers/creation/flowparser.py +++ b/parsers/creation/flowparser.py @@ -304,7 +304,7 @@ def parse(self): return flow_container def _parse_block(self, depth=0, block_type='root_block', omit_content=False): - row = self.sheet_parser.parse_next_row() + row = self.sheet_parser.parse_next_row(omit_templating=omit_content) while not self._is_end_of_block(block_type, row): if omit_content or not row.include_if: if row.type == 'begin_for': @@ -349,7 +349,7 @@ def _parse_block(self, depth=0, block_type='root_block', omit_content=False): self.append_node_group(new_node_group, row.row_id) else: self._parse_row(row) - row = self.sheet_parser.parse_next_row() + row = self.sheet_parser.parse_next_row(omit_templating=omit_content) def _is_end_of_block(self, block_type, row): block_end_map = {