diff --git a/muppet/puppet/ast.py b/muppet/puppet/ast.py index a8baa1a6b92f574086149d26710c7237dd370a26..028751dda1511518553d3cefd25b3079fc0db152 100644 --- a/muppet/puppet/ast.py +++ b/muppet/puppet/ast.py @@ -213,6 +213,7 @@ class PuppetUnless(Puppet): condition: Puppet consequent: list[Puppet] + alternative: list[Puppet] = field(default_factory=list) @dataclass @@ -679,10 +680,14 @@ def build_ast(form: Any) -> Puppet: return PuppetIfChain(clauses) + case ['unless', {'test': test, 'then': forms, 'else': else_}]: + return PuppetUnless(build_ast(test), + [build_ast(x) for x in forms], + [build_ast(x) for x in else_]) + case ['unless', {'test': test, 'then': forms}]: - return PuppetUnless(build_ast(test), [build_ast(x) for x in forms]) - case ['unless', {'test': test}]: - return PuppetUnless(build_ast(test), []) + return PuppetUnless(build_ast(test), + [build_ast(x) for x in forms]) case ['invoke', {'functor': func, 'args': args, 'block': block}]: return PuppetInvoke(build_ast(func), diff --git a/muppet/puppet/format/parser.py b/muppet/puppet/format/parser.py index ccb59f2711324f243e4648214717d41a65d29819..828e03b75860cdddb870f9394845c56c83c7686b 100644 --- a/muppet/puppet/format/parser.py +++ b/muppet/puppet/format/parser.py @@ -634,8 +634,11 @@ class ParserFormatter(Serializer[ParseDirective]): @override def _puppet_unless(self, it: PuppetUnless) -> ParseDirective: - return (ws & 'unless' & ws & self.s(it.condition) & ws & '{' & - ws & self.s(it.consequent) & ws & '}') + parser = (ws & 'unless' & ws & self.s(it.condition) & ws & '{' & + ws & self.s(it.consequent) & ws & '}') + parser &= optional(ws & 'else' & ws & '{' & ws & self.s(it.alternative) & + ws & '}') + return parser @override def _puppet_var(self, it: PuppetVar) -> ParseDirective: diff --git a/tests/test_parse_elsif.py b/tests/test_parse_elsif.py index 1bcf43dca815537a74fd74fc3a66fab71cd5fa13..02264153860141031a96640367f59feb815db72c 100644 --- a/tests/test_parse_elsif.py +++ b/tests/test_parse_elsif.py @@ -197,3 +197,16 @@ def test_collect_nested(): } } """) + + +def test_unless(): + # Outer if to force unless to parse correctly + parse_string(""" + if true { + unless 1 { + 2 + } else { + 3 + } + } + """)