7552552492
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
155 lines
5.5 KiB
Python
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')
|