deb-murano/murano/dsl/expressions.py
Stan Lagun 61f84d03ca Adds ability to throw/catch/rethrow exceptions in MuranoPL
The syntax is
Try:
  - Throw: ns:name   #can be list of names to simulate type hierarchy
     Message: message   #optional
     Cause: $sourceException   #optional
     Extra: { 'someExtra': 'data' }    #optional
Catch:
   - With: ns:name #can be list of names
   - As: exception   #optional
   - Do:
        - Rethrow:
Else:    #optional
    - else block
Finally:    #optional
     - finally block

Improves stack traces to contain information about Python native stack frames and macro blocks

Change-Id: I2e2bcc5e1a0da5f9489d73525f8b3fa99cc0220c
Implements: blueprint muranopl-exception-handling
2014-06-12 20:48:27 +04:00

113 lines
3.4 KiB
Python

# Copyright (c) 2014 Mirantis, Inc.
#
# Licensed under the Apache License, Version 2.0 (the "License"); you may
# not use this file except in compliance with the License. You may obtain
# a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
# License for the specific language governing permissions and limitations
# under the License.
import types
import murano.dsl.dsl_exception as dsl_exception
import murano.dsl.helpers as helpers
import murano.dsl.lhs_expression as lhs_expression
import murano.dsl.yaql_expression as yaql_expression
_macros = []
class InstructionStub(object):
def __init__(self, title, position):
self._title = title
self.source_file_position = position
def __str__(self):
return self._title
def register_macro(cls):
_macros.append(cls)
class DslExpression(object):
def execute(self, context, murano_class):
pass
class Statement(DslExpression):
def __init__(self, statement):
if isinstance(statement, yaql_expression.YaqlExpression):
key = None
value = statement
elif isinstance(statement, types.DictionaryType):
if len(statement) != 1:
raise SyntaxError()
key = statement.keys()[0]
value = statement[key]
else:
raise SyntaxError()
self._destination = lhs_expression.LhsExpression(key) if key else None
self._expression = value
@property
def destination(self):
return self._destination
@property
def expression(self):
return self._expression
def execute(self, context, murano_class):
try:
result = helpers.evaluate(self.expression, context)
if self.destination:
self.destination(result, context, murano_class)
return result
except dsl_exception.MuranoPlException:
raise
except Exception as e:
raise dsl_exception.MuranoPlException.from_python_exception(
e, context)
def parse_expression(expr):
result = None
if isinstance(expr, yaql_expression.YaqlExpression):
result = Statement(expr)
elif isinstance(expr, types.DictionaryType):
kwds = {}
for key, value in expr.iteritems():
if isinstance(key, yaql_expression.YaqlExpression):
if result is not None:
raise ValueError()
result = Statement(expr)
else:
kwds[key] = value
if result is None:
for cls in _macros:
try:
macro = cls(**kwds)
position = None
title = 'block construct'
if hasattr(expr, 'source_file_position'):
position = expr.source_file_position
if '__str__' in cls.__dict__:
title = str(macro)
macro.virtual_instruction = InstructionStub(
title, position)
return macro
except TypeError:
continue
if result is None:
raise SyntaxError()
return result