| import os |
| import pdb |
| import sys |
| import tempfile |
| sys.path.append("/opt/tosca") |
| from translator.toscalib.tosca_template import ToscaTemplate |
| |
| from core.models import Slice,Sliver,User,Flavor,Node,Image |
| from nodeselect import XOSNodeSelector |
| from imageselect import XOSImageSelector |
| |
| import resources |
| |
| class XOSTosca(object): |
| def __init__(self, tosca_yaml, parent_dir=None): |
| # 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() |
| |
| try: |
| (tmp_handle, tmp_pathname) = tempfile.mkstemp(dir=parent_dir) |
| os.write(tmp_handle, tosca_yaml) |
| os.close(tmp_handle) |
| |
| self.template = ToscaTemplate(tmp_pathname) |
| finally: |
| os.remove(tmp_pathname) |
| |
| self.compute_dependencies() |
| |
| self.ordered_nodetemplates = [] |
| self.ordered_names = self.topsort_dependencies() |
| print "ordered_names", 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 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) |
| |
| def execute_nodetemplate(self, user, nodetemplate): |
| if nodetemplate.type in resources.resources: |
| cls = resources.resources[nodetemplate.type] |
| print "work on", cls.__name__, nodetemplate.name |
| obj = cls(user, nodetemplate) |
| obj.create_or_update() |
| |
| 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) |
| for model in obj.get_existing_objs(): |
| models.append( (obj, model) ) |
| models.reverse() |
| for (resource,model) in models: |
| print "destroying", model |
| resource.delete(model) |
| |