murano/murano/dsl/murano_class.py

185 lines
5.9 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 collections
import inspect
import murano.dsl.exceptions as exceptions
import murano.dsl.helpers as helpers
import murano.dsl.murano_method as murano_method
import murano.dsl.murano_object as murano_object
import murano.dsl.typespec as typespec
def classname(name):
def wrapper(cls):
cls._murano_class_name = name
return cls
return wrapper
class MuranoClass(object):
def __init__(self, class_loader, namespace_resolver, name, package,
parents=None):
self._package = package
self._class_loader = class_loader
self._methods = {}
self._namespace_resolver = namespace_resolver
self._name = namespace_resolver.resolve_name(name)
self._properties = {}
if self._name == 'io.murano.Object':
self._parents = []
else:
self._parents = parents or [
class_loader.get_class('io.murano.Object')]
class_name = 'mc' + helpers.generate_id()
parents_class = [p.object_class for p in self._parents]
bases = tuple(parents_class) or (murano_object.MuranoObject,)
self.object_class = type(class_name, bases, {})
@property
def name(self):
return self._name
@property
def package(self):
return self._package
@property
def namespace_resolver(self):
return self._namespace_resolver
@property
def parents(self):
return self._parents
@property
def methods(self):
return self._methods
def get_method(self, name):
return self._methods.get(name)
def add_method(self, name, payload):
method = murano_method.MuranoMethod(self._namespace_resolver,
self, name, payload)
self._methods[name] = method
return method
@property
def properties(self):
return self._properties.keys()
def add_property(self, name, property_typespec):
if not isinstance(property_typespec, typespec.PropertySpec):
raise TypeError('property_typespec')
self._properties[name] = property_typespec
def get_property(self, name):
return self._properties[name]
def _find_method_chains(self, name):
initial = [self.methods[name]] if name in self.methods else []
yielded = False
for parent in self.parents:
for seq in parent._find_method_chains(name):
yield initial + list(seq)
yielded = True
if initial and not yielded:
yield initial
def find_single_method(self, name):
chains = sorted(self._find_method_chains(name), key=lambda t: len(t))
result = []
for i in range(len(chains)):
if chains[i][0] in result:
continue
add = True
for j in range(i + 1, len(chains)):
common = 0
if not add:
break
for p in range(len(chains[i])):
if chains[i][-p - 1] is chains[j][-p - 1]:
common += 1
else:
break
if common == len(chains[i]):
add = False
break
if add:
result.append(chains[i][0])
if len(result) < 1:
raise exceptions.NoMethodFound(name)
elif len(result) > 1:
raise exceptions.AmbiguousMethodName(name)
return result[0]
def find_all_methods(self, name):
result = []
queue = collections.deque([self])
while queue:
c = queue.popleft()
if name in c.methods:
method = c.methods[name]
if method not in result:
result.append(method)
queue.extend(c.parents)
return result
def find_property(self, name):
types = collections.deque([self])
while len(types) > 0:
mc = types.popleft()
if name in mc.properties:
return mc.get_property(name)
types.extend(mc.parents)
return None
def invoke(self, name, executor, this, parameters):
if not self.is_compatible(this):
raise Exception("'this' must be of compatible type")
args = executor.to_yaql_args(parameters)
return executor.invoke_method(name, this.cast(self), None, self, *args)
def is_compatible(self, obj):
if isinstance(obj, murano_object.MuranoObject):
return self.is_compatible(obj.type)
if obj is self:
return True
for parent in obj.parents:
if self.is_compatible(parent):
return True
return False
def new(self, parent, object_store, context, parameters=None,
object_id=None, **kwargs):
obj = self.object_class(self, parent, object_store, context,
object_id=object_id, **kwargs)
if parameters is not None:
argspec = inspect.getargspec(obj.initialize).args
if '_context' in argspec:
parameters['_context'] = context
if '_parent' in argspec:
parameters['_parent'] = parent
obj.initialize(**parameters)
return obj
def __str__(self):
return 'MuranoClass({0})'.format(self.name)