Serg Melikyan 7552552492 Add MuranoPL Engine
Added all missing parts to complete MuranoPL implementation:
 - system classes
 - base classes
 - integration with oslo.messaging
 - package entry-point & other things to run engine
Integrated engine with API

Note: some tests are marked to be skipped,
will be enabled via separate commit.

Partially-Implements: blueprint new-metadata-dsl
Change-Id: I3c1c2326b48da57647d55ea8edfba56f1657d7d6
2014-03-21 15:34:42 +04:00

155 lines
5.5 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.
from yaql import context
from muranoapi.engine import consts
from muranoapi.engine import exceptions
from muranoapi.engine import helpers
class MuranoObject(object):
def __init__(self, murano_class, parent_obj, object_store, context,
object_id=None, known_classes=None, defaults=None):
if known_classes is None:
known_classes = {}
self.__parent_obj = parent_obj
self.__object_id = object_id or helpers.generate_id()
self.__type = murano_class
self.__properties = {}
self.__object_store = object_store
self.__parents = {}
self.__context = context
self.__defaults = defaults or {}
known_classes[murano_class.name] = self
for parent_class in murano_class.parents:
name = parent_class.name
if not name in known_classes:
obj = parent_class.new(
parent_obj, object_store, context, None,
object_id=self.__object_id,
known_classes=known_classes,
defaults=defaults)
known_classes[name] = self.__parents[name] = obj
else:
self.__parents[name] = known_classes[name]
def initialize(self, **kwargs):
used_names = set()
for i in xrange(2):
for property_name in self.__type.properties:
spec = self.__type.get_property(property_name)
if i == 0 and helpers.needs_evaluation(spec.default) \
or i == 1 and property_name in used_names:
continue
used_names.add(property_name)
property_value = kwargs.get(property_name, consts.NoValue)
self.set_property(property_name, property_value)
for parent in self.__parents.values():
parent.initialize(**kwargs)
@property
def object_id(self):
return self.__object_id
@property
def type(self):
return self.__type
@property
def parent(self):
return self.__parent_obj
def __getattr__(self, item):
if item.startswith('__'):
raise AttributeError('Access to internal attributes is '
'restricted')
return self.get_property(item)
def get_property(self, item, caller_class=None):
try:
return self.__get_property(item, caller_class)
except AttributeError:
if not caller_class:
raise AttributeError(item)
try:
obj = self.cast(caller_class)
return obj.__properties[item]
except KeyError:
raise AttributeError(item)
except TypeError:
raise AttributeError(item)
def __get_property(self, item, caller_class=None):
if item in self.__properties:
return self.__properties[item]
i = 0
result = None
for parent in self.__parents.values():
try:
result = parent.__get_property(item, caller_class)
i += 1
if i > 1:
raise LookupError()
except AttributeError:
continue
if not i:
raise AttributeError()
return result
def set_property(self, key, value, caller_class=None):
try:
self.__set_property(key, value, caller_class)
except AttributeError as e:
if not caller_class:
raise e
try:
obj = self.cast(caller_class)
obj.__properties[key] = value
except TypeError:
raise AttributeError(key)
def __set_property(self, key, value, caller_class=None):
if key in self.__type.properties:
spec = self.__type.get_property(key)
if (caller_class is not None and
not caller_class.is_compatible(self)):
raise exceptions.NoWriteAccess(key)
default = self.__defaults.get(key, spec.default)
child_context = context.Context(parent_context=self.__context)
child_context.set_data(self)
default = helpers.evaluate(default, child_context, 1)
self.__properties[key] = spec.validate(
value, self, self.__context, self.__object_store, default)
else:
for parent in self.__parents.values():
try:
parent.__set_property(key, value, caller_class)
return
except AttributeError:
continue
raise AttributeError(key)
def cast(self, _type):
if self.type == _type:
return self
for parent in self.__parents.values():
try:
return parent.cast(_type)
except TypeError:
continue
raise TypeError('Cannot cast')