Fixing Bug 2103 and Bug 2134 in NBI
[osm/NBI.git] / osm_nbi / descriptor_topics.py
index 590380a..50182fd 100644 (file)
@@ -37,7 +37,12 @@ from osm_nbi.validation import (
     validate_input,
     vnfpkgop_new_schema,
 )
-from osm_nbi.base_topic import BaseTopic, EngineException, get_iterable
+from osm_nbi.base_topic import (
+    BaseTopic,
+    EngineException,
+    get_iterable,
+    detect_descriptor_usage,
+)
 from osm_im import etsi_nfv_vnfd, etsi_nfv_nsd
 from osm_im.nst import nst as nst_im
 from pyangbind.lib.serialise import pybindJSONDecoder
@@ -268,6 +273,8 @@ class DescriptorTopic(BaseTopic):
         # TODO change to Content-Disposition filename https://tools.ietf.org/html/rfc6266
         file_pkg = None
         error_text = ""
+        fs_rollback = []
+
         try:
             if content_range_text:
                 content_range = (
@@ -296,9 +303,10 @@ class DescriptorTopic(BaseTopic):
             else:
                 self.fs.file_delete(proposed_revision_path, ignore_non_exist=True)
                 self.fs.mkdir(proposed_revision_path)
+                fs_rollback.append(proposed_revision_path)
 
             storage = self.fs.get_params()
-            storage["folder"] = _id
+            storage["folder"] = proposed_revision_path
 
             file_path = (proposed_revision_path, filename)
             if self.fs.file_exists(file_path, "file"):
@@ -453,9 +461,11 @@ class DescriptorTopic(BaseTopic):
             if revision > 1:
                 try:
                     self._validate_descriptor_changes(
+                        _id,
                         descriptor_file_name,
                         current_revision_path,
-                        proposed_revision_path)
+                        proposed_revision_path,
+                    )
                 except Exception as e:
                     shutil.rmtree(self.fs.path + current_revision_path, ignore_errors=True)
                     shutil.rmtree(self.fs.path + proposed_revision_path, ignore_errors=True)
@@ -464,17 +474,6 @@ class DescriptorTopic(BaseTopic):
                     self.fs.file_delete(proposed_revision_path, ignore_non_exist=True)
                     raise e
 
-            # Copy the revision to the active package name by its original id
-            shutil.rmtree(self.fs.path + current_revision_path, ignore_errors=True)
-            os.rename(self.fs.path + proposed_revision_path, self.fs.path + current_revision_path)
-            self.fs.file_delete(current_revision_path, ignore_non_exist=True)
-            self.fs.mkdir(current_revision_path)
-            self.fs.reverse_sync(from_path=current_revision_path)
-            shutil.rmtree(self.fs.path + _id)
-
-            current_desc["_admin"]["storage"] = storage
-            current_desc["_admin"]["onboardingState"] = "ONBOARDED"
-            current_desc["_admin"]["operationalState"] = "ENABLED"
 
             indata = self._remove_envelop(indata)
 
@@ -482,18 +481,33 @@ class DescriptorTopic(BaseTopic):
             if kwargs:
                 self._update_input_with_kwargs(indata, kwargs)
 
+            current_desc["_admin"]["storage"] = storage
+            current_desc["_admin"]["onboardingState"] = "ONBOARDED"
+            current_desc["_admin"]["operationalState"] = "ENABLED"
+            current_desc["_admin"]["modified"] = time()
+            current_desc["_admin"]["revision"] = revision
+
             deep_update_rfc7396(current_desc, indata)
             current_desc = self.check_conflict_on_edit(
                 session, current_desc, indata, _id=_id
             )
-            current_desc["_admin"]["modified"] = time()
-            current_desc["_admin"]["revision"] = revision
+
+            # Copy the revision to the active package name by its original id
+            shutil.rmtree(self.fs.path + current_revision_path, ignore_errors=True)
+            os.rename(self.fs.path + proposed_revision_path, self.fs.path + current_revision_path)
+            self.fs.file_delete(current_revision_path, ignore_non_exist=True)
+            self.fs.mkdir(current_revision_path)
+            self.fs.reverse_sync(from_path=current_revision_path)
+
+            shutil.rmtree(self.fs.path + _id)
+
             self.db.replace(self.topic, _id, current_desc)
 
             #  Store a copy of the package as a point in time revision
             revision_desc = dict(current_desc)
             revision_desc["_id"] = _id + ":" + str(revision_desc["_admin"]["revision"])
             self.db.create(self.topic + "_revisions", revision_desc)
+            fs_rollback = []
 
             indata["_id"] = _id
             self._send_msg("edited", indata)
@@ -525,6 +539,8 @@ class DescriptorTopic(BaseTopic):
         finally:
             if file_pkg:
                 file_pkg.close()
+            for file in fs_rollback:
+                self.fs.file_delete(file, ignore_non_exist=True)
 
     def get_file(self, session, _id, path=None, accept_header=None):
         """
@@ -559,7 +575,7 @@ class DescriptorTopic(BaseTopic):
             )
         storage = content["_admin"]["storage"]
         if path is not None and path != "$DESCRIPTOR":  # artifacts
-            if not storage.get("pkg-dir"):
+            if not storage.get("pkg-dir") and not storage.get("folder"):
                 raise EngineException(
                     "Packages does not contains artifacts",
                     http_code=HTTPStatus.BAD_REQUEST,
@@ -685,11 +701,13 @@ class DescriptorTopic(BaseTopic):
 
         return indata
 
-    def _validate_descriptor_changes(self,
+    def _validate_descriptor_changes(
+        self,
+        descriptor_id,
         descriptor_file_name,
         old_descriptor_directory,
-        new_descriptor_directory):
-        # Todo: compare changes and throw a meaningful exception for the user to understand
+        new_descriptor_directory
+    ):
         # Example:
         #    raise EngineException(
         #           "Error in validating new descriptor: <NODE> cannot be modified",
@@ -1214,7 +1232,14 @@ class VnfdTopic(DescriptorTopic):
         Returns:
             vnfd (dict): VNFD which does not include policies
         """
-        # TODO: Extract the policy related parts from the VNFD
+        for df in vnfd.get("df", {}):
+            for policy in ["scaling-aspect", "healing-aspect"]:
+                if (df.get(policy, {})):
+                    df.pop(policy)
+        for vdu in vnfd.get("vdu", {}):
+            for alarm_policy in ["alarm", "monitoring-parameter"]:
+                if (vdu.get(alarm_policy, {})):
+                    vdu.pop(alarm_policy)
         return vnfd
 
     @staticmethod
@@ -1274,6 +1299,7 @@ class VnfdTopic(DescriptorTopic):
 
     def _validate_descriptor_changes(
         self,
+        descriptor_id: str,
         descriptor_file_name: str,
         old_descriptor_directory: str,
         new_descriptor_directory: str,
@@ -1282,7 +1308,7 @@ class VnfdTopic(DescriptorTopic):
 
         Args:
             old_descriptor_directory (str):   Directory of descriptor which is in-use
-            new_descriptor_directory (str):   Directory of directory which is proposed to update (new revision)
+            new_descriptor_directory (str):   Directory of descriptor which is proposed to update (new revision)
 
         Returns:
             None
@@ -1291,27 +1317,37 @@ class VnfdTopic(DescriptorTopic):
             EngineException:    In case of error when there are unallowed changes
         """
         try:
+            # If VNFD does not exist in DB or it is not in use by any NS,
+            # validation is not required.
+            vnfd = self.db.get_one("vnfds", {"_id": descriptor_id})
+            if not vnfd or not detect_descriptor_usage(vnfd, "vnfds", self.db):
+                return
+
+            # Get the old and new descriptor contents in order to compare them.
             with self.fs.file_open(
                 (old_descriptor_directory.rstrip("/"), descriptor_file_name), "r"
             ) as old_descriptor_file:
+
                 with self.fs.file_open(
-                    (new_descriptor_directory, descriptor_file_name), "r"
+                    (new_descriptor_directory.rstrip("/"), descriptor_file_name), "r"
                 ) as new_descriptor_file:
-                    old_content = yaml.load(
-                        old_descriptor_file.read(), Loader=yaml.SafeLoader
-                    )
-                    new_content = yaml.load(
-                        new_descriptor_file.read(), Loader=yaml.SafeLoader
-                    )
+
+                    old_content = yaml.safe_load(old_descriptor_file.read())
+                    new_content = yaml.safe_load(new_descriptor_file.read())
+
+                    # If software version has changed, we do not need to validate
+                    # the differences anymore.
                     if old_content and new_content:
                         if self.find_software_version(
                             old_content
                         ) != self.find_software_version(new_content):
                             return
+
                         disallowed_change = DeepDiff(
                             self.remove_modifiable_items(old_content),
                             self.remove_modifiable_items(new_content),
                         )
+
                         if disallowed_change:
                             changed_nodes = functools.reduce(
                                 lambda a, b: a + " , " + b,
@@ -1322,6 +1358,7 @@ class VnfdTopic(DescriptorTopic):
                                     ).keys()
                                 ],
                             )
+
                             raise EngineException(
                                 f"Error in validating new descriptor: {changed_nodes} cannot be modified, "
                                 "there are disallowed changes in the vnf descriptor.",
@@ -1657,6 +1694,7 @@ class NsdTopic(DescriptorTopic):
 
     def _validate_descriptor_changes(
         self,
+        descriptor_id: str,
         descriptor_file_name: str,
         old_descriptor_directory: str,
         new_descriptor_directory: str,
@@ -1665,7 +1703,7 @@ class NsdTopic(DescriptorTopic):
 
         Args:
             old_descriptor_directory:   Directory of descriptor which is in-use
-            new_descriptor_directory:   Directory of directory which is proposed to update (new revision)
+            new_descriptor_directory:   Directory of descriptor which is proposed to update (new revision)
 
         Returns:
             None
@@ -1675,23 +1713,30 @@ class NsdTopic(DescriptorTopic):
         """
 
         try:
+            # If NSD does not exist in DB, or it is not in use by any NS,
+            # validation is not required.
+            nsd = self.db.get_one("nsds", {"_id": descriptor_id}, fail_on_empty=False)
+            if not nsd or not detect_descriptor_usage(nsd, "nsds", self.db):
+                return
+
+            # Get the old and new descriptor contents in order to compare them.
             with self.fs.file_open(
-                (old_descriptor_directory, descriptor_file_name), "r"
+                (old_descriptor_directory.rstrip("/"), descriptor_file_name), "r"
             ) as old_descriptor_file:
+
                 with self.fs.file_open(
                     (new_descriptor_directory.rstrip("/"), descriptor_file_name), "r"
                 ) as new_descriptor_file:
-                    old_content = yaml.load(
-                        old_descriptor_file.read(), Loader=yaml.SafeLoader
-                    )
-                    new_content = yaml.load(
-                        new_descriptor_file.read(), Loader=yaml.SafeLoader
-                    )
+
+                    old_content = yaml.safe_load(old_descriptor_file.read())
+                    new_content = yaml.safe_load(new_descriptor_file.read())
+
                     if old_content and new_content:
                         disallowed_change = DeepDiff(
                             self.remove_modifiable_items(old_content),
                             self.remove_modifiable_items(new_content),
                         )
+
                         if disallowed_change:
                             changed_nodes = functools.reduce(
                                 lambda a, b: a + ", " + b,
@@ -1702,6 +1747,7 @@ class NsdTopic(DescriptorTopic):
                                     ).keys()
                                 ],
                             )
+
                             raise EngineException(
                                 f"Error in validating new descriptor: {changed_nodes} cannot be modified, "
                                 "there are disallowed changes in the ns descriptor. ",