From 2441427941c8f310e025da406cd524a7a4a211a1 Mon Sep 17 00:00:00 2001 From: Shkelqim Memolla Date: Mon, 16 Dec 2024 15:28:25 +0100 Subject: [PATCH 1/2] fix: fail on unmatched tags In the case there are unmatched tags, after the search is done, the parsing should fail. --- pystache/parser.py | 9 ++++----- pystache/tests/test_parser.py | 36 ++++++++++++++++++++++++++++++++--- 2 files changed, 37 insertions(+), 8 deletions(-) diff --git a/pystache/parser.py b/pystache/parser.py index 30fd1756..938dda9e 100644 --- a/pystache/parser.py +++ b/pystache/parser.py @@ -237,17 +237,12 @@ def _change_delimiters(self, delimiters): def parse(self, template): """ Parse a template string starting at some index. - This method uses the current tag delimiter. Arguments: - template: a unicode string that is the template to parse. - index: the index at which to start parsing. - Returns: - a ParsedTemplate instance. """ @@ -337,6 +332,10 @@ def parse(self, template): parsed_template.add(node) + # Some open/close tags were mismatched. + if states: + raise ParsingError('Tag mismatch.') + # Avoid adding spurious empty strings to the parse tree. if start_index != len(template): parsed_template.add(template[start_index:]) diff --git a/pystache/tests/test_parser.py b/pystache/tests/test_parser.py index 1e3c9012..11853b55 100644 --- a/pystache/tests/test_parser.py +++ b/pystache/tests/test_parser.py @@ -8,19 +8,49 @@ import unittest from pystache.defaults import DELIMITERS -from pystache.parser import _compile_template_re as make_re +from pystache.parser import _compile_template_re as make_re, parse, ParsingError class RegularExpressionTestCase(unittest.TestCase): - """Tests the regular expression returned by _compile_template_re().""" def test_re(self): """ Test getting a key from a dictionary. - """ re = make_re(DELIMITERS) match = re.search("b {{test}}") self.assertEqual(match.start(), 1) + + +class ParseTestCase(unittest.TestCase): + """Tests the parse() function.""" + + def test_parse_okay(self): + """ + Test parsing templates in the cases there are no errors. + """ + ts = [ + '
{{>A}}
', + '{{^A}}
some text
{{/A}}', + '{{#A}} {{^B}} {{/B}} {{/A}}', + '{{#A}} {{^B}} {{/B}} {{/A}} {{#C}} {{/C}}', + ] + for t in ts: + with self.subTest(template=t): + parse(t) + + def test_parse_fail(self): + """ + Test parsing templates in the cases there are errors. + """ + ts = [ + '{{#A}}
some text
', + '{{#A}}
some text
{{/A}}
TEXT
{{/B}}', + '{{#A}} {{#B}} {{/A}} {{/B}}', + ] + for t in ts: + with self.subTest(template=t): + with self.assertRaises(ParsingError): + parse(t) From 1d82da5de3e30ebb837064e772f9797ebc49f1da Mon Sep 17 00:00:00 2001 From: Shkelqim Memolla Date: Fri, 20 Dec 2024 14:39:11 +0100 Subject: [PATCH 2/2] chore: add raise_on_mismatch parameter In the normal behaviour, if there's a tag mismatch the parse function, will run succesfully. In case the this marked as True, then the parse will raise an exception. This is intended for stricter parsing. --- pystache/parser.py | 13 +++++++------ pystache/tests/test_parser.py | 3 ++- 2 files changed, 9 insertions(+), 7 deletions(-) diff --git a/pystache/parser.py b/pystache/parser.py index 938dda9e..a2d3fc42 100644 --- a/pystache/parser.py +++ b/pystache/parser.py @@ -17,15 +17,14 @@ # TODO: add some unit tests for this. # TODO: add a test case that checks for spurious spaces. # TODO: add test cases for delimiters. -def parse(template, delimiters=None): +def parse(template, delimiters=None, raise_on_mismatch=False): """ Parse a unicode template string and return a ParsedTemplate instance. Arguments: - template: a unicode template string. - delimiters: a 2-tuple of delimiters. Defaults to the package default. + raise_on_mismatch: a boolean indicating whether to raise an exception when parsing fails. Examples: @@ -36,7 +35,7 @@ def parse(template, delimiters=None): """ if type(template) is not str: raise Exception('Template is not unicode: %s' % type(template)) - parser = _Parser(delimiters) + parser = _Parser(delimiters, raise_on_mismatch=raise_on_mismatch) return parser.parse(template) @@ -220,12 +219,14 @@ def render(self, engine, context): class _Parser(object): _delimiters = None _template_re = None + _raise_on_mismatch = False - def __init__(self, delimiters=None): + def __init__(self, delimiters=None, raise_on_mismatch=False): if delimiters is None: delimiters = defaults.DELIMITERS self._delimiters = delimiters + self._raise_on_mismatch = raise_on_mismatch def _compile_delimiters(self): self._template_re = _compile_template_re(self._delimiters) @@ -333,7 +334,7 @@ def parse(self, template): parsed_template.add(node) # Some open/close tags were mismatched. - if states: + if self._raise_on_mismatch and states: raise ParsingError('Tag mismatch.') # Avoid adding spurious empty strings to the parse tree. diff --git a/pystache/tests/test_parser.py b/pystache/tests/test_parser.py index 11853b55..e376d785 100644 --- a/pystache/tests/test_parser.py +++ b/pystache/tests/test_parser.py @@ -33,6 +33,7 @@ def test_parse_okay(self): """ ts = [ '
{{>A}}
', + '{{#A}}
some text
', '{{^A}}
some text
{{/A}}', '{{#A}} {{^B}} {{/B}} {{/A}}', '{{#A}} {{^B}} {{/B}} {{/A}} {{#C}} {{/C}}', @@ -53,4 +54,4 @@ def test_parse_fail(self): for t in ts: with self.subTest(template=t): with self.assertRaises(ParsingError): - parse(t) + parse(t, raise_on_mismatch=True)