From 1ebda2688ac0898c7e673646568ae6255104506c Mon Sep 17 00:00:00 2001
From: Jorrit Schaap <schaap@astron.nl>
Date: Tue, 11 Feb 2020 13:22:30 +0100
Subject: [PATCH] TMSS-142: initial implementation of validation of subtask
 specification against its schema

---
 .../src/tmss/tmssapp/models/scheduling.py     | 23 ++++++++++++++++++-
 1 file changed, 22 insertions(+), 1 deletion(-)

diff --git a/SAS/TMSS/src/tmss/tmssapp/models/scheduling.py b/SAS/TMSS/src/tmss/tmssapp/models/scheduling.py
index 4a0c2d19106..a693066284a 100644
--- a/SAS/TMSS/src/tmss/tmssapp/models/scheduling.py
+++ b/SAS/TMSS/src/tmss/tmssapp/models/scheduling.py
@@ -8,6 +8,13 @@ from django.contrib.postgres.fields import ArrayField, JSONField
 from .specification import AbstractChoice, BasicCommon, Template, NamedCommon # , <TaskBlueprint
 from enum import Enum
 from rest_framework.serializers import HyperlinkedRelatedField
+
+from lofar.sas.tmss.tmss.exceptions import *
+
+import json
+import fastjsonschema
+import jsonschema
+
 #
 # I/O
 #
@@ -127,7 +134,6 @@ class DataproductFeedbackTemplate(Template):
 #
 # Instance Objects
 #
-
 class Subtask(BasicCommon):
     """
     Represents a low-level task, which is an atomic unit of execution, such as running an observation, running
@@ -147,6 +153,21 @@ class Subtask(BasicCommon):
     scheduler_input_doc = JSONField(help_text='Partial specifications, as input for the scheduler.')
     # resource_claim = ForeignKey("ResourceClaim", null=False, on_delete=PROTECT) # todo <-- how is this external reference supposed to work?
 
+    def validate_specification_against_schema(self):
+        try:
+            # ensure the specification and schema are both valid json in the first place
+            spec = json.loads(self.specifications_doc) if type(self.specifications_doc) == str else self.specifications_doc
+            schema = json.loads(self.specifications_template.schema) if type(self.specifications_template.schema) == str else self.specifications_template.schema
+            jsonschema.validate(spec, schema)
+        except Exception as e:
+            raise SpecificationException(str(e))
+
+    def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
+        '''override of normal save method, doing a validation of the specification against the schema first
+        :raises SpecificationException in case the specification does not validate against the schema'''
+        self.validate_specification_against_schema()
+        super().save(force_insert, force_update, using, update_fields)
+
 
 class SubtaskInput(BasicCommon):
     subtask = ForeignKey('Subtask', null=False, on_delete=CASCADE, help_text='Subtask to which this input specification refers.')
-- 
GitLab