From 9cb7caad31d6f7c0f7adc394b367c4c74b683c3c Mon Sep 17 00:00:00 2001
From: Jorrit Schaap <schaap@astron.nl>
Date: Thu, 4 Nov 2021 11:35:48 +0100
Subject: [PATCH] TMSS-1144: made dataproducts and referenced model(parts)
 immutable when a subtask is finished

---
 .../migrations/0011_dataproduct_immutable.py  | 137 ++++++++++++++++++
 .../src/tmss/tmssapp/models/scheduling.py     |   4 +
 2 files changed, 141 insertions(+)
 create mode 100644 SAS/TMSS/backend/src/tmss/tmssapp/migrations/0011_dataproduct_immutable.py

diff --git a/SAS/TMSS/backend/src/tmss/tmssapp/migrations/0011_dataproduct_immutable.py b/SAS/TMSS/backend/src/tmss/tmssapp/migrations/0011_dataproduct_immutable.py
new file mode 100644
index 00000000000..078bb74e512
--- /dev/null
+++ b/SAS/TMSS/backend/src/tmss/tmssapp/migrations/0011_dataproduct_immutable.py
@@ -0,0 +1,137 @@
+# Generated by Django 3.0.9 on 2021-10-07 09:17
+
+from django.db import migrations, models
+import django.db.models.deletion
+
+
+class Migration(migrations.Migration):
+
+    dependencies = [
+        ('tmssapp', '0010_subtaskinputoutput'),
+    ]
+
+    operations = [
+        migrations.RunSQL('''CREATE OR REPLACE FUNCTION tmssapp_block_dataproduct_immutable_fields_update()
+                      RETURNS trigger AS
+                      $BODY$
+                      DECLARE
+                        dp_producing_subtask_state tmssapp_subtaskstate.value%type;
+                      BEGIN
+                        SELECT tmssapp_subtask.state_id FROM tmssapp_dataproduct INTO dp_producing_subtask_state
+                        INNER JOIN tmssapp_subtaskoutput on tmssapp_subtaskoutput.id=tmssapp_dataproduct.producer_id
+                        INNER JOIN tmssapp_subtask on tmssapp_subtask.id=tmssapp_subtaskoutput.subtask_id
+                        WHERE tmssapp_dataproduct.id=OLD.id
+                        LIMIT 1;
+
+                        IF dp_producing_subtask_state = 'finished' THEN
+                          IF OLD.filename <> NEW.filename OR
+                             OLD.directory <> NEW.directory OR
+                             OLD.size <> NEW.size OR
+                             OLD.dataformat_id <> NEW.dataformat_id OR
+                             OLD.datatype_id <> NEW.datatype_id OR
+                             OLD.specifications_doc <> NEW.specifications_doc OR
+                             OLD.specifications_template_id <> NEW.specifications_template_id OR
+                             OLD.feedback_doc <> NEW.feedback_doc OR
+                             OLD.feedback_template_id <> NEW.feedback_template_id OR
+                             OLD.sap_id <> NEW.sap_id OR
+                             OLD.global_identifier_id <> NEW.global_identifier_id OR
+                             OLD.producer_id <> NEW.producer_id THEN
+                            RAISE EXCEPTION 'ILLEGAL UPDATE OF IMMUTABLE DATAPRODUCT FIELD';
+                          END IF;
+                        END IF;
+                      RETURN NEW;
+                      END;
+                      $BODY$
+                      LANGUAGE plpgsql VOLATILE;
+                      DROP TRIGGER IF EXISTS tmssapp_block_dataproduct_immutable_fields_update ON tmssapp_dataproduct ;
+                      CREATE TRIGGER tmssapp_block_dataproduct_immutable_fields_update
+                      BEFORE UPDATE ON tmssapp_dataproduct
+                      FOR EACH ROW EXECUTE PROCEDURE tmssapp_block_dataproduct_immutable_fields_update();
+                      '''),
+        migrations.RunSQL('''CREATE OR REPLACE FUNCTION tmssapp_block_dataproducthash_immutable_fields_update()
+                  RETURNS trigger AS
+                  $BODY$
+                  DECLARE
+                    dp_producing_subtask_state tmssapp_subtaskstate.value%type;
+                  BEGIN
+                    SELECT tmssapp_subtask.state_id FROM tmssapp_dataproducthash INTO dp_producing_subtask_state
+                    INNER JOIN tmssapp_dataproduct on tmssapp_dataproduct.id=tmssapp_dataproducthash.dataproduct_id
+                    INNER JOIN tmssapp_subtaskoutput on tmssapp_subtaskoutput.id=tmssapp_dataproduct.producer_id
+                    INNER JOIN tmssapp_subtask on tmssapp_subtask.id=tmssapp_subtaskoutput.subtask_id
+                    WHERE tmssapp_dataproducthash.id=OLD.id
+                    LIMIT 1;
+
+                    IF dp_producing_subtask_state = 'finished' THEN
+                      IF OLD.hash <> NEW.hash OR
+                         OLD.hash_algorithm_id <> NEW.hash_algorithm_id OR
+                         OLD.dataproduct_id <> NEW.dataproduct_id THEN
+                        RAISE EXCEPTION 'ILLEGAL UPDATE OF IMMUTABLE DATAPRODUCTHASH FIELD';
+                      END IF;
+                    END IF;
+                  RETURN NEW;
+                  END;
+                  $BODY$
+                  LANGUAGE plpgsql VOLATILE;
+                  DROP TRIGGER IF EXISTS tmssapp_block_dataproducthash_immutable_fields_update ON tmssapp_dataproducthash ;
+                  CREATE TRIGGER tmssapp_block_dataproducthash_immutable_fields_update
+                  BEFORE UPDATE ON tmssapp_dataproducthash
+                  FOR EACH ROW EXECUTE PROCEDURE tmssapp_block_dataproducthash_immutable_fields_update();
+                  '''),
+        migrations.RunSQL('''CREATE OR REPLACE FUNCTION tmssapp_block_dataproductarchiveinfo_immutable_fields_update()
+              RETURNS trigger AS
+              $BODY$
+              DECLARE
+                dp_producing_subtask_state tmssapp_subtaskstate.value%type;
+              BEGIN
+                SELECT tmssapp_subtask.state_id FROM tmssapp_dataproductarchiveinfo INTO dp_producing_subtask_state
+                INNER JOIN tmssapp_dataproduct on tmssapp_dataproduct.id=tmssapp_dataproductarchiveinfo.dataproduct_id
+                INNER JOIN tmssapp_subtaskoutput on tmssapp_subtaskoutput.id=tmssapp_dataproduct.producer_id
+                INNER JOIN tmssapp_subtask on tmssapp_subtask.id=tmssapp_subtaskoutput.subtask_id
+                WHERE tmssapp_dataproductarchiveinfo.id=OLD.id
+                LIMIT 1;
+
+                IF dp_producing_subtask_state = 'finished' THEN
+                  IF OLD.storage_ticket <> NEW.storage_ticket OR
+                     OLD.dataproduct_id <> NEW.dataproduct_id THEN
+                    RAISE EXCEPTION 'ILLEGAL UPDATE OF IMMUTABLE DATAPRODUCTARCHIVEINFO FIELD';
+                  END IF;
+                END IF;
+              RETURN NEW;
+              END;
+              $BODY$
+              LANGUAGE plpgsql VOLATILE;
+              DROP TRIGGER IF EXISTS tmssapp_block_dataproductarchiveinfo_immutable_fields_update ON tmssapp_dataproductarchiveinfo ;
+              CREATE TRIGGER tmssapp_block_dataproductarchiveinfo_immutable_fields_update
+              BEFORE UPDATE ON tmssapp_dataproductarchiveinfo
+              FOR EACH ROW EXECUTE PROCEDURE tmssapp_block_dataproductarchiveinfo_immutable_fields_update();
+              '''),
+        migrations.RunSQL('''CREATE OR REPLACE FUNCTION tmssapp_block_sap_immutable_fields_update()
+              RETURNS trigger AS
+              $BODY$
+              DECLARE
+                dp_producing_subtask_state tmssapp_subtaskstate.value%type;
+              BEGIN
+                SELECT tmssapp_subtask.state_id FROM tmssapp_sap INTO dp_producing_subtask_state
+                INNER JOIN tmssapp_dataproduct on tmssapp_dataproduct.sap_id=tmssapp_sap.id
+                INNER JOIN tmssapp_subtaskoutput on tmssapp_subtaskoutput.id=tmssapp_dataproduct.producer_id
+                INNER JOIN tmssapp_subtask on tmssapp_subtask.id=tmssapp_subtaskoutput.subtask_id
+                WHERE tmssapp_sap.id=OLD.id
+                LIMIT 1;
+
+                IF dp_producing_subtask_state = 'finished' THEN
+                  IF OLD.specifications_doc <> NEW.specifications_doc OR
+                     OLD.specifications_template_id <> NEW.specifications_template_id OR
+                     OLD.global_identifier_id <> NEW.global_identifier_id THEN
+                    RAISE EXCEPTION 'ILLEGAL UPDATE OF IMMUTABLE SAP FIELD';
+                  END IF;
+                END IF;
+              RETURN NEW;
+              END;
+              $BODY$
+              LANGUAGE plpgsql VOLATILE;
+              DROP TRIGGER IF EXISTS tmssapp_block_sap_immutable_fields_update ON tmssapp_sap ;
+              CREATE TRIGGER tmssapp_block_sap_immutable_fields_update
+              BEFORE UPDATE ON tmssapp_sap
+              FOR EACH ROW EXECUTE PROCEDURE tmssapp_block_sap_immutable_fields_update();
+              ''')
+    ]
diff --git a/SAS/TMSS/backend/src/tmss/tmssapp/models/scheduling.py b/SAS/TMSS/backend/src/tmss/tmssapp/models/scheduling.py
index 71e988bfb99..728953c7f39 100644
--- a/SAS/TMSS/backend/src/tmss/tmssapp/models/scheduling.py
+++ b/SAS/TMSS/backend/src/tmss/tmssapp/models/scheduling.py
@@ -443,6 +443,7 @@ class SAP(BasicCommon, TemplateSchemaMixin):
     specifications_doc = JSONField(help_text='SAP properties.')
     specifications_template = ForeignKey('SAPTemplate', null=False, on_delete=CASCADE, help_text='Schema used for specifications_doc.')
     global_identifier = OneToOneField('SIPidentifier', null=False, editable=False, on_delete=PROTECT, help_text='The global unique identifier for LTA SIP.')
+    # please note that the all fields become immutable when its dataproduct's producing subtask is finished (See SQL triggers)
 
     def save(self, force_insert=False, force_update=False, using=None, update_fields=None):
         self.annotate_validate_add_defaults_to_doc_using_template('specifications_doc', 'specifications_template')
@@ -472,6 +473,7 @@ class Dataproduct(TemplateSchemaMixin, RefreshFromDbInvalidatesCachedPropertiesM
     feedback_template = ForeignKey('DataproductFeedbackTemplate', on_delete=PROTECT, help_text='Schema used for feedback_doc.')
     sap = ForeignKey('SAP', on_delete=PROTECT, null=True, related_name="dataproducts", help_text='SAP this dataproduct was generated out of (NULLable).')
     global_identifier = OneToOneField('SIPidentifier', editable=False, null=False, on_delete=PROTECT, help_text='The global unique identifier for LTA SIP.')
+    # please note that the all fields except deleted_since and expected_size become immutable when its dataproduct's producing subtask is finished (See SQL triggers)
 
     class Meta(BasicCommon.Meta):
         constraints = [UniqueConstraint(fields=['directory', 'filename'], name='%(class)s_unique_path')]
@@ -525,10 +527,12 @@ class DataproductArchiveInfo(BasicCommon):
     storage_ticket = CharField(max_length=128, help_text='Archive-system identifier.')
     public_since = DateTimeField(null=True, help_text='Dataproduct is available for public download since this moment, or NULL if dataproduct is not (NULLable).')
     corrupted_since = DateTimeField(null=True, help_text='Earliest timestamp from which this dataproduct is known to be partially or fully corrupt, or NULL if dataproduct is not known to be corrupt (NULLable).')
+    # please note that the dataproduct, storage_ticket fields become immutable when its dataproduct's producing subtask is finished (See SQL triggers)
 
 
 class DataproductHash(BasicCommon):
     dataproduct = ForeignKey('Dataproduct', related_name='hashes', on_delete=PROTECT, help_text='The dataproduct to which this hash refers.')
     hash_algorithm = ForeignKey('HashAlgorithm', null=False, on_delete=PROTECT, help_text='Algorithm used for hashing (MD5, AES256).')
     hash = CharField(max_length=128, help_text='Hash value.')
+    # please note that the dataproduct, hash_algorithm, hash fields become immutable when its dataproduct's producing subtask is finished (See SQL triggers)
 
-- 
GitLab