2020-03-27 12:29:31 +00:00
|
|
|
import json
|
|
|
|
import unittest
|
2020-05-07 17:57:24 +00:00
|
|
|
from unittest.mock import patch
|
2020-03-27 12:29:31 +00:00
|
|
|
|
|
|
|
from crc import session
|
|
|
|
from crc.api.common import ApiErrorSchema
|
|
|
|
from crc.models.file import FileModel
|
2020-05-07 17:57:24 +00:00
|
|
|
from crc.models.protocol_builder import ProtocolBuilderStudySchema
|
2020-03-27 12:29:31 +00:00
|
|
|
from crc.models.workflow import WorkflowSpecModel, WorkflowSpecModelSchema, WorkflowModel, WorkflowSpecCategoryModel
|
|
|
|
from tests.base_test import BaseTest
|
|
|
|
|
|
|
|
|
|
|
|
class TestWorkflowSpecValidation(BaseTest):
|
|
|
|
|
|
|
|
def validate_workflow(self, workflow_name):
|
|
|
|
self.load_example_data()
|
|
|
|
spec_model = self.load_test_spec(workflow_name)
|
|
|
|
rv = self.app.get('/v1.0/workflow-specification/%s/validate' % spec_model.id, headers=self.logged_in_headers())
|
|
|
|
self.assert_success(rv)
|
|
|
|
json_data = json.loads(rv.get_data(as_text=True))
|
|
|
|
return ApiErrorSchema(many=True).load(json_data)
|
|
|
|
|
2020-05-22 18:37:49 +00:00
|
|
|
def test_successful_validation_of_test_workflows(self):
|
2020-05-07 17:57:24 +00:00
|
|
|
|
2020-03-27 12:29:31 +00:00
|
|
|
self.assertEqual(0, len(self.validate_workflow("parallel_tasks")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("decision_table")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("docx")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("exclusive_gateway")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("file_upload_form")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("random_fact")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("study_details")))
|
|
|
|
self.assertEqual(0, len(self.validate_workflow("two_forms")))
|
|
|
|
|
2020-05-07 17:57:24 +00:00
|
|
|
@patch('crc.services.protocol_builder.ProtocolBuilderService.get_investigators') # mock_studies
|
|
|
|
@patch('crc.services.protocol_builder.ProtocolBuilderService.get_required_docs') # mock_docs
|
|
|
|
@patch('crc.services.protocol_builder.ProtocolBuilderService.get_study_details') # mock_details
|
|
|
|
@patch('crc.services.protocol_builder.ProtocolBuilderService.get_studies') # mock_studies
|
|
|
|
def test_successful_validation_of_auto_loaded_workflows(self, mock_studies, mock_details, mock_docs, mock_investigators):
|
|
|
|
|
|
|
|
# Mock Protocol Builder responses
|
|
|
|
studies_response = self.protocol_builder_response('user_studies.json')
|
|
|
|
mock_studies.return_value = ProtocolBuilderStudySchema(many=True).loads(studies_response)
|
|
|
|
details_response = self.protocol_builder_response('study_details.json')
|
|
|
|
mock_details.return_value = json.loads(details_response)
|
|
|
|
docs_response = self.protocol_builder_response('required_docs.json')
|
|
|
|
mock_docs.return_value = json.loads(docs_response)
|
|
|
|
investigators_response = self.protocol_builder_response('investigators.json')
|
|
|
|
mock_investigators.return_value = json.loads(investigators_response)
|
|
|
|
|
2020-03-27 12:29:31 +00:00
|
|
|
self.load_example_data()
|
|
|
|
workflows = session.query(WorkflowSpecModel).all()
|
|
|
|
errors = []
|
|
|
|
for w in workflows:
|
|
|
|
rv = self.app.get('/v1.0/workflow-specification/%s/validate' % w.id,
|
|
|
|
headers=self.logged_in_headers())
|
|
|
|
self.assert_success(rv)
|
|
|
|
json_data = json.loads(rv.get_data(as_text=True))
|
|
|
|
errors.extend(ApiErrorSchema(many=True).load(json_data))
|
|
|
|
self.assertEqual(0, len(errors), json.dumps(errors))
|
|
|
|
|
|
|
|
def test_invalid_expression(self):
|
|
|
|
errors = self.validate_workflow("invalid_expression")
|
|
|
|
self.assertEqual(1, len(errors))
|
2020-05-11 21:04:05 +00:00
|
|
|
self.assertEqual("workflow_execution_exception", errors[0]['code'])
|
2020-04-06 22:15:32 +00:00
|
|
|
self.assertEqual("ExclusiveGateway_003amsm", errors[0]['task_id'])
|
|
|
|
self.assertEqual("Has Bananas Gateway", errors[0]['task_name'])
|
|
|
|
self.assertEqual("invalid_expression.bpmn", errors[0]['file_name'])
|
2020-05-11 21:04:05 +00:00
|
|
|
self.assertEqual('ExclusiveGateway_003amsm: Error evaluating expression \'this_value_does_not_exist==true\', '
|
|
|
|
'name \'this_value_does_not_exist\' is not defined', errors[0]["message"])
|
2020-03-27 12:29:31 +00:00
|
|
|
|
|
|
|
def test_validation_error(self):
|
|
|
|
errors = self.validate_workflow("invalid_spec")
|
|
|
|
self.assertEqual(1, len(errors))
|
2020-04-06 22:15:32 +00:00
|
|
|
self.assertEqual("workflow_validation_error", errors[0]['code'])
|
|
|
|
self.assertEqual("StartEvent_1", errors[0]['task_id'])
|
|
|
|
self.assertEqual("invalid_spec.bpmn", errors[0]['file_name'])
|
2020-03-27 12:29:31 +00:00
|
|
|
|
|
|
|
def test_invalid_script(self):
|
|
|
|
errors = self.validate_workflow("invalid_script")
|
|
|
|
self.assertEqual(1, len(errors))
|
2020-04-06 22:15:32 +00:00
|
|
|
self.assertEqual("workflow_execution_exception", errors[0]['code'])
|
2020-03-27 12:29:31 +00:00
|
|
|
self.assertTrue("NoSuchScript" in errors[0]['message'])
|
2020-04-06 22:15:32 +00:00
|
|
|
self.assertEqual("Invalid_Script_Task", errors[0]['task_id'])
|
|
|
|
self.assertEqual("An Invalid Script Reference", errors[0]['task_name'])
|
|
|
|
self.assertEqual("invalid_script.bpmn", errors[0]['file_name'])
|