f49bdb6218
Since python 2.2, use of types module is not prefered way to access the type hence it is replaced by using names of built-in factory functions. Change-Id: Ibdcc1ae75204f67076edab5edcffe94a9f1f1d8e Implements: blueprint replace-types-module-with-new-style-classes
170 lines
5.7 KiB
Python
170 lines
5.7 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 utils
|
|
|
|
from murano.dsl import dsl
|
|
from murano.dsl import dsl_types
|
|
from murano.dsl import murano_method
|
|
|
|
|
|
class ObjRef(object):
|
|
def __init__(self, obj):
|
|
self.ref_obj = obj
|
|
|
|
|
|
def serialize(obj):
|
|
return serialize_model(obj, None, True)['Objects']
|
|
|
|
|
|
def _serialize_object(root_object, designer_attributes, allow_refs):
|
|
serialized_objects = set()
|
|
|
|
obj = root_object
|
|
while True:
|
|
obj, need_another_pass = _pass12_serialize(
|
|
obj, None, serialized_objects, designer_attributes)
|
|
if not need_another_pass:
|
|
break
|
|
tree = [obj]
|
|
_pass3_serialize(tree, serialized_objects, allow_refs)
|
|
return tree[0], serialized_objects
|
|
|
|
|
|
def serialize_model(root_object, executor, allow_refs=False):
|
|
if executor is not None:
|
|
designer_attributes = executor.object_store.designer_attributes
|
|
else:
|
|
designer_attributes = None
|
|
|
|
if root_object is None:
|
|
tree = None
|
|
tree_copy = None
|
|
attributes = []
|
|
else:
|
|
tree, serialized_objects = _serialize_object(
|
|
root_object, designer_attributes, allow_refs)
|
|
tree_copy, _ = _serialize_object(root_object, None, allow_refs)
|
|
if executor is not None:
|
|
attributes = executor.attribute_store.serialize(serialized_objects)
|
|
else:
|
|
attributes = []
|
|
|
|
return {
|
|
'Objects': tree,
|
|
'ObjectsCopy': tree_copy,
|
|
'Attributes': attributes
|
|
}
|
|
|
|
|
|
def _serialize_available_action(obj, current_actions):
|
|
result = {}
|
|
actions = obj.type.find_methods(
|
|
lambda m: m.usage == murano_method.MethodUsages.Action)
|
|
for action in actions:
|
|
action_id = '{0}_{1}'.format(obj.object_id, action.name)
|
|
entry = current_actions.get(action_id, {'enabled': True})
|
|
entry['name'] = action.name
|
|
result[action_id] = entry
|
|
return result
|
|
|
|
|
|
def _pass12_serialize(value, parent, serialized_objects,
|
|
designer_attributes_getter):
|
|
if isinstance(value, dsl.MuranoObjectInterface):
|
|
value = value.object
|
|
if isinstance(value, (basestring, int, float, bool)) or value is None:
|
|
return value, False
|
|
if isinstance(value, dsl_types.MuranoObject):
|
|
if value.owner is not parent or value.object_id in serialized_objects:
|
|
return ObjRef(value), True
|
|
elif isinstance(value, ObjRef):
|
|
if (value.ref_obj.object_id not in serialized_objects and
|
|
is_nested_in(value.ref_obj.owner, parent)):
|
|
value = value.ref_obj
|
|
else:
|
|
return value, False
|
|
if isinstance(value, dsl_types.MuranoObject):
|
|
result = value.to_dictionary()
|
|
if designer_attributes_getter is not None:
|
|
result['?'].update(designer_attributes_getter(value.object_id))
|
|
# deserialize and merge list of actions
|
|
result['?']['_actions'] = _serialize_available_action(
|
|
value, result['?'].get('_actions', {}))
|
|
serialized_objects.add(value.object_id)
|
|
return _pass12_serialize(
|
|
result, value, serialized_objects, designer_attributes_getter)
|
|
elif isinstance(value, utils.MappingType):
|
|
result = {}
|
|
need_another_pass = False
|
|
|
|
for d_key, d_value in value.iteritems():
|
|
result_key = str(d_key)
|
|
result_value = _pass12_serialize(
|
|
d_value, parent, serialized_objects,
|
|
designer_attributes_getter)
|
|
result[result_key] = result_value[0]
|
|
if result_value[1]:
|
|
need_another_pass = True
|
|
return result, need_another_pass
|
|
elif utils.is_sequence(value) or isinstance(value, utils.SetType):
|
|
need_another_pass = False
|
|
result = []
|
|
for t in value:
|
|
v, nmp = _pass12_serialize(
|
|
t, parent, serialized_objects, designer_attributes_getter)
|
|
if nmp:
|
|
need_another_pass = True
|
|
result.append(v)
|
|
return result, need_another_pass
|
|
else:
|
|
raise ValueError()
|
|
|
|
|
|
def _pass3_serialize(value, serialized_objects, allow_refs=False):
|
|
if isinstance(value, dict):
|
|
for d_key, d_value in value.items():
|
|
if isinstance(d_value, ObjRef):
|
|
if (d_value.ref_obj.object_id in serialized_objects or
|
|
allow_refs):
|
|
value[d_key] = d_value.ref_obj.object_id
|
|
else:
|
|
del value[d_key]
|
|
else:
|
|
_pass3_serialize(d_value, serialized_objects, allow_refs)
|
|
elif isinstance(value, list):
|
|
index = 0
|
|
while index < len(value):
|
|
item = value[index]
|
|
if isinstance(item, ObjRef):
|
|
if item.ref_obj.object_id in serialized_objects or allow_refs:
|
|
value[index] = item.ref_obj.object_id
|
|
else:
|
|
value.pop(index)
|
|
index -= 1
|
|
else:
|
|
_pass3_serialize(item, serialized_objects, allow_refs)
|
|
index += 1
|
|
return value
|
|
|
|
|
|
def is_nested_in(obj, ancestor):
|
|
while True:
|
|
if obj is ancestor:
|
|
return True
|
|
if obj is None:
|
|
return False
|
|
obj = obj.owner
|