blob: 12ab3f5a8e56a2226a73b8ac2546df14b1ef09cd [file] [log] [blame]
import os
import pdb
import resources
import sys
import tempfile
import traceback
from toscaparser.tosca_template import ToscaTemplate
from core.models import Slice,Instance,User,Flavor,Node,Image
from nodeselect import XOSNodeSelector
from imageselect import XOSImageSelector
class XOSTosca(object):
def __init__(self, tosca_yaml, parent_dir=None, log_to_console = False):
# TOSCA will look for imports using a relative path from where the
# template file is located, so we have to put the template file
# in a specific place.
if not parent_dir:
parent_dir = os.getcwd()
tmp_pathname = None
try:
(tmp_handle, tmp_pathname) = tempfile.mkstemp(dir=parent_dir, suffix=".yaml")
os.write(tmp_handle, tosca_yaml)
os.close(tmp_handle)
self.template = ToscaTemplate(tmp_pathname)
except:
traceback.print_exc()
raise
finally:
if tmp_pathname:
os.remove(tmp_pathname)
self.log_to_console = log_to_console
self.log_msgs = []
self.compute_dependencies()
self.deferred_sync = []
self.ordered_nodetemplates = []
self.ordered_names = self.topsort_dependencies()
self.log("ordered_names: %s" % self.ordered_names)
for name in self.ordered_names:
if name in self.nodetemplates_by_name:
self.ordered_nodetemplates.append(self.nodetemplates_by_name[name])
#pdb.set_trace()
def log(self, msg):
if self.log_to_console:
print msg
self.log_msgs.append(msg)
def compute_dependencies(self):
nodetemplates_by_name = {}
for nodetemplate in self.template.nodetemplates:
nodetemplates_by_name[nodetemplate.name] = nodetemplate
self.nodetemplates_by_name = nodetemplates_by_name
for nodetemplate in self.template.nodetemplates:
nodetemplate.dependencies = []
nodetemplate.dependencies_names = []
for reqs in nodetemplate.requirements:
for (k,v) in reqs.items():
name = v["node"]
if (name in nodetemplates_by_name):
nodetemplate.dependencies.append(nodetemplates_by_name[name])
nodetemplate.dependencies_names.append(name)
# go another level deep, as our requirements can have requirements...
for sd_req in v.get("requirements",[]):
for (sd_req_k, sd_req_v) in sd_req.items():
name = sd_req_v["node"]
if (name in nodetemplates_by_name):
nodetemplate.dependencies.append(nodetemplates_by_name[name])
nodetemplate.dependencies_names.append(name)
def topsort_dependencies(self):
# stolen from observer
g = self.nodetemplates_by_name
# Get set of all nodes, including those without outgoing edges
keys = set(g.keys())
values = set({})
for v in g.values():
values=values | set(v.dependencies_names)
all_nodes=list(keys|values)
steps = all_nodes
# Final order
order = []
# DFS stack, not using recursion
stack = []
# Unmarked set
unmarked = all_nodes
# visiting = [] - skip, don't expect 1000s of nodes, |E|/|V| is small
while unmarked:
stack.insert(0,unmarked[0]) # push first unmarked
while (stack):
n = stack[0]
add = True
try:
for m in g[n].dependencies_names:
if (m in unmarked):
add = False
stack.insert(0,m)
except KeyError:
pass
if (add):
if (n in steps and n not in order):
order.append(n)
item = stack.pop(0)
try:
unmarked.remove(item)
except ValueError:
pass
noorder = list(set(steps) - set(order))
return order + noorder
def execute(self, user):
for nodetemplate in self.ordered_nodetemplates:
self.execute_nodetemplate(user, nodetemplate)
for obj in self.deferred_sync:
self.log("Saving deferred sync obj %s" % obj)
obj.no_sync = False
obj.save()
def execute_nodetemplate(self, user, nodetemplate):
if nodetemplate.type not in resources.resources:
raise Exception("Nodetemplate %s's type %s is not a known resource" % (nodetemplate.name, nodetemplate.type))
cls = resources.resources[nodetemplate.type]
#print "work on", cls.__name__, nodetemplate.name
obj = cls(user, nodetemplate, self)
obj.create_or_update()
self.deferred_sync = self.deferred_sync + obj.deferred_sync
def destroy(self, user):
nodetemplates = self.ordered_nodetemplates
models = []
for nodetemplate in nodetemplates:
if nodetemplate.type in resources.resources:
cls = resources.resources[nodetemplate.type]
obj = cls(user, nodetemplate, self)
for model in obj.get_existing_objs():
models.append( (obj, model) )
models.reverse()
for (resource,model) in models:
resource.delete(model)
def name_to_xos_class(self, user, name):
nt = self.nodetemplates_by_name.get(name)
if not nt:
raise Exception("failed to find nodetemplate %s" % name)
cls = resources.resources.get(nt.type)
if not cls:
raise Exception("nodetemplate %s's type does not resolve to a known resource type" % name)
return (nt, cls, cls.xos_model)
def name_to_xos_model(self, user, name):
(nt, cls, model_class) = self.name_to_xos_class(user, name)
obj = cls(user, nt, self)
existing_objs = obj.get_existing_objs()
if not existing_objs:
raise Exception("failed to find xos %s %s" % (cls.__name__, name))
return existing_objs[0]